maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 1 | // Copyright 2015 The Chromium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
maxbogue | 455a57e3 | 2016-08-14 00:08:32 | [diff] [blame] | 5 | #include "components/sync/driver/sync_stopped_reporter.h" |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 6 | |
Sebastien Marchand | 53801a3 | 2019-01-25 16:26:11 | [diff] [blame] | 7 | #include "base/bind.h" |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 8 | #include "base/run_loop.h" |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 9 | #include "base/test/bind_test_util.h" |
gab | a6f7232 | 2017-03-02 00:46:14 | [diff] [blame] | 10 | #include "base/test/scoped_mock_time_message_loop_task_runner.h" |
Gabriel Charette | c710874 | 2019-08-23 03:31:40 | [diff] [blame^] | 11 | #include "base/test/task_environment.h" |
Max Bogue | fef332d | 2016-07-28 22:09:09 | [diff] [blame] | 12 | #include "components/sync/protocol/sync.pb.h" |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 13 | #include "net/http/http_status_code.h" |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 14 | #include "net/http/http_util.h" |
| 15 | #include "services/network/public/cpp/resource_request.h" |
| 16 | #include "services/network/public/cpp/shared_url_loader_factory.h" |
| 17 | #include "services/network/public/cpp/weak_wrapper_shared_url_loader_factory.h" |
| 18 | #include "services/network/test/test_url_loader_factory.h" |
John Abd-El-Malek | 2c58b14 | 2018-06-28 14:40:30 | [diff] [blame] | 19 | #include "services/network/test/test_utils.h" |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 20 | #include "testing/gtest/include/gtest/gtest.h" |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 21 | |
maxbogue | 7e006db | 2016-10-03 19:48:28 | [diff] [blame] | 22 | namespace syncer { |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 23 | |
| 24 | const char kTestURL[] = "http://chromium.org/test"; |
| 25 | const char kTestURLTrailingSlash[] = "http://chromium.org/test/"; |
| 26 | const char kEventURL[] = "http://chromium.org/test/event"; |
| 27 | |
maxbogue | b18b920 | 2015-08-11 17:52:12 | [diff] [blame] | 28 | const char kTestUserAgent[] = "the_fifth_element"; |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 29 | const char kAuthToken[] = "multipass"; |
| 30 | const char kCacheGuid[] = "leeloo"; |
| 31 | const char kBirthday[] = "2263"; |
| 32 | |
maxbogue | c5c836a | 2015-06-15 22:16:38 | [diff] [blame] | 33 | const char kAuthHeaderPrefix[] = "Bearer "; |
| 34 | |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 35 | class SyncStoppedReporterTest : public testing::Test { |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 36 | protected: |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 37 | SyncStoppedReporterTest() {} |
| 38 | ~SyncStoppedReporterTest() override {} |
| 39 | |
| 40 | void SetUp() override { |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 41 | test_shared_loader_factory_ = |
| 42 | base::MakeRefCounted<network::WeakWrapperSharedURLLoaderFactory>( |
| 43 | url_loader_factory()); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 44 | } |
| 45 | |
| 46 | void RequestFinishedCallback(const SyncStoppedReporter::Result& result) { |
| 47 | request_result_ = result; |
| 48 | } |
| 49 | |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 50 | GURL interception_url(const GURL& url) { |
| 51 | return SyncStoppedReporter::GetSyncEventURL(url); |
| 52 | } |
| 53 | |
maxbogue | 455a57e3 | 2016-08-14 00:08:32 | [diff] [blame] | 54 | GURL test_url() { return GURL(kTestURL); } |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 55 | |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 56 | void call_on_timeout(SyncStoppedReporter* ssr) { ssr->OnTimeout(); } |
| 57 | |
maxbogue | 455a57e3 | 2016-08-14 00:08:32 | [diff] [blame] | 58 | std::string user_agent() const { return std::string(kTestUserAgent); } |
maxbogue | b18b920 | 2015-08-11 17:52:12 | [diff] [blame] | 59 | |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 60 | SyncStoppedReporter::ResultCallback callback() { |
| 61 | return base::Bind(&SyncStoppedReporterTest::RequestFinishedCallback, |
| 62 | base::Unretained(this)); |
| 63 | } |
| 64 | |
| 65 | const SyncStoppedReporter::Result& request_result() const { |
| 66 | return request_result_; |
| 67 | } |
| 68 | |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 69 | network::TestURLLoaderFactory* url_loader_factory() { |
| 70 | return &test_url_loader_factory_; |
| 71 | } |
| 72 | |
| 73 | scoped_refptr<network::SharedURLLoaderFactory> shared_url_loader_factory() { |
| 74 | return test_shared_loader_factory_; |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 75 | } |
| 76 | |
| 77 | private: |
Gabriel Charette | dfa3604 | 2019-08-19 17:30:11 | [diff] [blame] | 78 | base::test::TaskEnvironment task_environment_; |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 79 | network::TestURLLoaderFactory test_url_loader_factory_; |
| 80 | scoped_refptr<network::SharedURLLoaderFactory> test_shared_loader_factory_; |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 81 | SyncStoppedReporter::Result request_result_; |
| 82 | |
| 83 | DISALLOW_COPY_AND_ASSIGN(SyncStoppedReporterTest); |
| 84 | }; |
| 85 | |
| 86 | // Test that the event URL gets constructed correctly. |
| 87 | TEST_F(SyncStoppedReporterTest, EventURL) { |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 88 | GURL intercepted_url; |
| 89 | url_loader_factory()->AddResponse(interception_url(GURL(kTestURL)).spec(), |
| 90 | ""); |
| 91 | url_loader_factory()->SetInterceptor( |
| 92 | base::BindLambdaForTesting([&](const network::ResourceRequest& request) { |
| 93 | intercepted_url = request.url; |
| 94 | })); |
| 95 | SyncStoppedReporter ssr(GURL(kTestURL), user_agent(), |
| 96 | shared_url_loader_factory(), callback()); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 97 | ssr.ReportSyncStopped(kAuthToken, kCacheGuid, kBirthday); |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 98 | EXPECT_EQ(kEventURL, intercepted_url.spec()); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 99 | } |
| 100 | |
| 101 | // Test that the event URL gets constructed correctly with a trailing slash. |
| 102 | TEST_F(SyncStoppedReporterTest, EventURLWithSlash) { |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 103 | GURL intercepted_url; |
| 104 | url_loader_factory()->AddResponse( |
| 105 | interception_url(GURL(kTestURLTrailingSlash)).spec(), ""); |
| 106 | url_loader_factory()->SetInterceptor( |
| 107 | base::BindLambdaForTesting([&](const network::ResourceRequest& request) { |
| 108 | intercepted_url = request.url; |
| 109 | })); |
maxbogue | b18b920 | 2015-08-11 17:52:12 | [diff] [blame] | 110 | SyncStoppedReporter ssr(GURL(kTestURLTrailingSlash), user_agent(), |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 111 | shared_url_loader_factory(), callback()); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 112 | ssr.ReportSyncStopped(kAuthToken, kCacheGuid, kBirthday); |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 113 | EXPECT_EQ(kEventURL, intercepted_url.spec()); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 114 | } |
| 115 | |
| 116 | // Test that the URLFetcher gets configured correctly. |
| 117 | TEST_F(SyncStoppedReporterTest, FetcherConfiguration) { |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 118 | GURL intercepted_url; |
| 119 | net::HttpRequestHeaders intercepted_headers; |
| 120 | std::string intercepted_body; |
| 121 | url_loader_factory()->AddResponse(interception_url(test_url()).spec(), ""); |
| 122 | url_loader_factory()->SetInterceptor( |
| 123 | base::BindLambdaForTesting([&](const network::ResourceRequest& request) { |
| 124 | intercepted_url = request.url; |
| 125 | intercepted_headers = request.headers; |
John Abd-El-Malek | 2c58b14 | 2018-06-28 14:40:30 | [diff] [blame] | 126 | intercepted_body = network::GetUploadData(request); |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 127 | })); |
| 128 | SyncStoppedReporter ssr(test_url(), user_agent(), shared_url_loader_factory(), |
maxbogue | 455a57e3 | 2016-08-14 00:08:32 | [diff] [blame] | 129 | callback()); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 130 | ssr.ReportSyncStopped(kAuthToken, kCacheGuid, kBirthday); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 131 | |
maxbogue | c5c836a | 2015-06-15 22:16:38 | [diff] [blame] | 132 | // Ensure the headers are set correctly. |
maxbogue | c5c836a | 2015-06-15 22:16:38 | [diff] [blame] | 133 | std::string header; |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 134 | intercepted_headers.GetHeader(net::HttpRequestHeaders::kAuthorization, |
| 135 | &header); |
maxbogue | c5c836a | 2015-06-15 22:16:38 | [diff] [blame] | 136 | std::string auth_header(kAuthHeaderPrefix); |
| 137 | auth_header.append(kAuthToken); |
| 138 | EXPECT_EQ(auth_header, header); |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 139 | intercepted_headers.GetHeader(net::HttpRequestHeaders::kUserAgent, &header); |
maxbogue | b18b920 | 2015-08-11 17:52:12 | [diff] [blame] | 140 | EXPECT_EQ(user_agent(), header); |
maxbogue | c5c836a | 2015-06-15 22:16:38 | [diff] [blame] | 141 | |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 142 | sync_pb::EventRequest event_request; |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 143 | event_request.ParseFromString(intercepted_body); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 144 | |
| 145 | EXPECT_EQ(kCacheGuid, event_request.sync_disabled().cache_guid()); |
| 146 | EXPECT_EQ(kBirthday, event_request.sync_disabled().store_birthday()); |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 147 | EXPECT_EQ(kEventURL, intercepted_url.spec()); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 148 | } |
| 149 | |
| 150 | TEST_F(SyncStoppedReporterTest, HappyCase) { |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 151 | url_loader_factory()->AddResponse(interception_url(test_url()).spec(), ""); |
| 152 | SyncStoppedReporter ssr(test_url(), user_agent(), shared_url_loader_factory(), |
maxbogue | 455a57e3 | 2016-08-14 00:08:32 | [diff] [blame] | 153 | callback()); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 154 | ssr.ReportSyncStopped(kAuthToken, kCacheGuid, kBirthday); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 155 | base::RunLoop run_loop; |
| 156 | run_loop.RunUntilIdle(); |
| 157 | EXPECT_EQ(SyncStoppedReporter::RESULT_SUCCESS, request_result()); |
| 158 | } |
| 159 | |
| 160 | TEST_F(SyncStoppedReporterTest, ServerNotFound) { |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 161 | url_loader_factory()->AddResponse(interception_url(test_url()).spec(), "", |
| 162 | net::HTTP_NOT_FOUND); |
| 163 | SyncStoppedReporter ssr(test_url(), user_agent(), shared_url_loader_factory(), |
maxbogue | 455a57e3 | 2016-08-14 00:08:32 | [diff] [blame] | 164 | callback()); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 165 | ssr.ReportSyncStopped(kAuthToken, kCacheGuid, kBirthday); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 166 | base::RunLoop run_loop; |
| 167 | run_loop.RunUntilIdle(); |
| 168 | EXPECT_EQ(SyncStoppedReporter::RESULT_ERROR, request_result()); |
| 169 | } |
| 170 | |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 171 | TEST_F(SyncStoppedReporterTest, Timeout) { |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 172 | url_loader_factory()->AddResponse(interception_url(test_url()).spec(), ""); |
gab | a6f7232 | 2017-03-02 00:46:14 | [diff] [blame] | 173 | // Mock the underlying loop's clock to trigger the timer at will. |
| 174 | base::ScopedMockTimeMessageLoopTaskRunner mock_main_runner; |
| 175 | |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 176 | SyncStoppedReporter ssr(test_url(), user_agent(), shared_url_loader_factory(), |
maxbogue | 455a57e3 | 2016-08-14 00:08:32 | [diff] [blame] | 177 | callback()); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 178 | |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 179 | // Begin request. |
| 180 | ssr.ReportSyncStopped(kAuthToken, kCacheGuid, kBirthday); |
| 181 | |
| 182 | // Trigger the timeout. |
gab | a6f7232 | 2017-03-02 00:46:14 | [diff] [blame] | 183 | ASSERT_TRUE(mock_main_runner->HasPendingTask()); |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 184 | call_on_timeout(&ssr); |
gab | a6f7232 | 2017-03-02 00:46:14 | [diff] [blame] | 185 | mock_main_runner->FastForwardUntilNoTasksRemain(); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 186 | EXPECT_EQ(SyncStoppedReporter::RESULT_TIMEOUT, request_result()); |
| 187 | } |
| 188 | |
| 189 | TEST_F(SyncStoppedReporterTest, NoCallback) { |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 190 | url_loader_factory()->AddResponse(interception_url(GURL(kTestURL)).spec(), |
| 191 | ""); |
| 192 | SyncStoppedReporter ssr(GURL(kTestURL), user_agent(), |
| 193 | shared_url_loader_factory(), |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 194 | SyncStoppedReporter::ResultCallback()); |
| 195 | ssr.ReportSyncStopped(kAuthToken, kCacheGuid, kBirthday); |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 196 | base::RunLoop run_loop; |
| 197 | run_loop.RunUntilIdle(); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 198 | } |
| 199 | |
| 200 | TEST_F(SyncStoppedReporterTest, NoCallbackTimeout) { |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 201 | url_loader_factory()->AddResponse(interception_url(GURL(kTestURL)).spec(), |
| 202 | ""); |
gab | a6f7232 | 2017-03-02 00:46:14 | [diff] [blame] | 203 | // Mock the underlying loop's clock to trigger the timer at will. |
| 204 | base::ScopedMockTimeMessageLoopTaskRunner mock_main_runner; |
| 205 | |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 206 | SyncStoppedReporter ssr(GURL(kTestURL), user_agent(), |
| 207 | shared_url_loader_factory(), |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 208 | SyncStoppedReporter::ResultCallback()); |
| 209 | |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 210 | // Begin request. |
| 211 | ssr.ReportSyncStopped(kAuthToken, kCacheGuid, kBirthday); |
| 212 | |
| 213 | // Trigger the timeout. |
gab | a6f7232 | 2017-03-02 00:46:14 | [diff] [blame] | 214 | ASSERT_TRUE(mock_main_runner->HasPendingTask()); |
Mark Pilgrim | 215ccf9 | 2018-06-11 22:32:19 | [diff] [blame] | 215 | call_on_timeout(&ssr); |
gab | a6f7232 | 2017-03-02 00:46:14 | [diff] [blame] | 216 | mock_main_runner->FastForwardUntilNoTasksRemain(); |
maxbogue | 334611e | 2015-02-20 04:08:17 | [diff] [blame] | 217 | } |
maxbogue | 7e006db | 2016-10-03 19:48:28 | [diff] [blame] | 218 | |
| 219 | } // namespace syncer |