Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "base/run_loop.h" | |
| 6 #include "google_apis/gaia/google_service_auth_error.h" | |
| 7 #include "net/url_request/test_url_fetcher_factory.h" | |
| 8 #include "net/url_request/url_request_test_util.h" | |
| 5 #include "sync/notifier/gcm_network_channel.h" | 9 #include "sync/notifier/gcm_network_channel.h" |
| 6 | |
| 7 #include "base/compiler_specific.h" | |
| 8 #include "testing/gtest/include/gtest/gtest.h" | 10 #include "testing/gtest/include/gtest/gtest.h" |
| 9 | 11 |
| 10 namespace syncer { | 12 namespace syncer { |
| 11 namespace { | 13 namespace { |
| 12 | 14 |
| 15 class TestGCMNetworkChannelDelegate : public GCMNetworkChannelDelegate { | |
| 16 public: | |
| 17 virtual void RequestToken(RequestTokenCallback callback) OVERRIDE { | |
| 18 request_token_callback = callback; | |
| 19 } | |
| 20 | |
| 21 virtual void InvalidateToken(const std::string& token) OVERRIDE { | |
| 22 invalidated_token = token; | |
| 23 } | |
| 24 | |
| 25 virtual void Register(RegisterCallback callback) OVERRIDE { | |
| 26 register_callback = callback; | |
| 27 } | |
| 28 | |
| 29 RequestTokenCallback request_token_callback; | |
| 30 std::string invalidated_token; | |
| 31 RegisterCallback register_callback; | |
| 32 }; | |
| 33 | |
| 13 class GCMNetworkChannelTest | 34 class GCMNetworkChannelTest |
|
tim (not reviewing)
2014/01/17 19:33:42
nice tests!
| |
| 14 : public ::testing::Test, | 35 : public ::testing::Test, |
| 15 public SyncNetworkChannel::Observer { | 36 public SyncNetworkChannel::Observer { |
| 16 protected: | 37 protected: |
| 17 GCMNetworkChannelTest() | 38 GCMNetworkChannelTest() |
| 18 : gcm_network_channel_() { | 39 : url_fetchers_created_count_(0) { |
|
tim (not reviewing)
2014/01/17 19:33:42
nit - init |delegate_| to NULL.
pavely
2014/01/17 22:52:26
Done.
| |
| 19 gcm_network_channel_.AddObserver(this); | |
| 20 gcm_network_channel_.SetMessageReceiver( | |
| 21 invalidation::NewPermanentCallback( | |
| 22 this, &GCMNetworkChannelTest::OnIncomingMessage)); | |
| 23 } | 40 } |
| 24 | 41 |
| 25 virtual ~GCMNetworkChannelTest() { | 42 virtual ~GCMNetworkChannelTest() { |
| 26 gcm_network_channel_.RemoveObserver(this); | 43 } |
| 44 | |
| 45 virtual void SetUp() { | |
| 46 request_context_getter_ = new net::TestURLRequestContextGetter( | |
| 47 base::MessageLoopProxy::current()); | |
| 48 // Ownership of delegate goes to GCNMentworkChannel but test needs pointer | |
| 49 // to it. | |
| 50 delegate_ = new TestGCMNetworkChannelDelegate(); | |
| 51 scoped_ptr<GCMNetworkChannelDelegate> delegate(delegate_); | |
| 52 gcm_network_channel_.reset(new GCMNetworkChannel(request_context_getter_, | |
| 53 delegate.Pass())); | |
| 54 gcm_network_channel_->AddObserver(this); | |
| 55 gcm_network_channel_->SetMessageReceiver( | |
| 56 invalidation::NewPermanentCallback( | |
| 57 this, &GCMNetworkChannelTest::OnIncomingMessage)); | |
| 58 url_fetcher_factory_.reset(new net::FakeURLFetcherFactory(NULL, | |
| 59 base::Bind(&GCMNetworkChannelTest::CreateURLFetcher, | |
| 60 base::Unretained(this)))); | |
| 61 } | |
| 62 | |
| 63 virtual void TearDown() { | |
| 64 gcm_network_channel_->RemoveObserver(this); | |
| 27 } | 65 } |
| 28 | 66 |
| 29 virtual void OnNetworkChannelStateChanged( | 67 virtual void OnNetworkChannelStateChanged( |
| 30 InvalidatorState invalidator_state) OVERRIDE { | 68 InvalidatorState invalidator_state) OVERRIDE { |
| 31 } | 69 } |
| 32 | 70 |
| 33 void OnIncomingMessage(std::string incoming_message) { | 71 void OnIncomingMessage(std::string incoming_message) { |
| 34 } | 72 } |
| 35 | 73 |
| 36 GCMNetworkChannel gcm_network_channel_; | 74 GCMNetworkChannel* network_channel() { |
| 75 return gcm_network_channel_.get(); | |
| 76 } | |
| 77 | |
| 78 TestGCMNetworkChannelDelegate* delegate() { | |
| 79 return delegate_; | |
| 80 } | |
| 81 | |
| 82 int url_fetchers_created_count() { | |
| 83 return url_fetchers_created_count_; | |
| 84 } | |
| 85 | |
| 86 net::FakeURLFetcherFactory* url_fetcher_factory() { | |
| 87 return url_fetcher_factory_.get(); | |
| 88 } | |
| 89 | |
| 90 scoped_ptr<net::FakeURLFetcher> CreateURLFetcher( | |
| 91 const GURL& url, | |
| 92 net::URLFetcherDelegate* delegate, | |
| 93 const std::string& response_data, | |
| 94 net::HttpStatusCode response_code, | |
| 95 net::URLRequestStatus::Status status) { | |
| 96 url_fetchers_created_count_++; | |
| 97 return scoped_ptr<net::FakeURLFetcher>(new net::FakeURLFetcher( | |
| 98 url, delegate, response_data, response_code, status)); | |
| 99 } | |
| 100 | |
| 101 private: | |
| 102 base::MessageLoop message_loop_; | |
| 103 TestGCMNetworkChannelDelegate* delegate_; | |
| 104 scoped_ptr<GCMNetworkChannel> gcm_network_channel_; | |
| 105 scoped_refptr<net::TestURLRequestContextGetter> request_context_getter_; | |
| 106 scoped_ptr<net::FakeURLFetcherFactory> url_fetcher_factory_; | |
| 107 int url_fetchers_created_count_; | |
| 37 }; | 108 }; |
| 38 | 109 |
| 110 TEST_F(GCMNetworkChannelTest, HappyCase) { | |
| 111 GURL url("http://invalid.url.com"); | |
| 112 url_fetcher_factory()->SetFakeResponse(url, "", net::HTTP_OK, | |
|
tim (not reviewing)
2014/01/17 19:33:42
nit - prefer std::string to "".
pavely
2014/01/17 22:52:26
Done.
| |
| 113 net::URLRequestStatus::SUCCESS); | |
| 114 | |
| 115 // After construction GCMNetworkChannel should have called Register. | |
| 116 EXPECT_FALSE(delegate()->register_callback.is_null()); | |
| 117 // Return valid registration id. | |
| 118 delegate()->register_callback.Run("registration.id", gcm::GCMClient::SUCCESS); | |
| 119 | |
| 120 network_channel()->SendMessage("abra.cadabra"); | |
| 121 // SendMessage should have triggered RequestToken. No HTTP request should be | |
| 122 // started yet. | |
| 123 EXPECT_FALSE(delegate()->request_token_callback.is_null()); | |
| 124 EXPECT_EQ(url_fetchers_created_count(), 0); | |
| 125 // Return valid access token. This should trigger HTTP request. | |
| 126 delegate()->request_token_callback.Run( | |
| 127 GoogleServiceAuthError::AuthErrorNone(), "access.token"); | |
| 128 { | |
| 129 base::RunLoop run_loop; | |
| 130 run_loop.RunUntilIdle(); | |
| 131 } | |
| 132 EXPECT_EQ(url_fetchers_created_count(), 1); | |
| 133 | |
| 134 // Return another access token. Message should be cleared by now and shouldn't | |
| 135 // be sent. | |
| 136 delegate()->request_token_callback.Run( | |
| 137 GoogleServiceAuthError::AuthErrorNone(), "access.token2"); | |
| 138 { | |
| 139 base::RunLoop run_loop; | |
| 140 run_loop.RunUntilIdle(); | |
| 141 } | |
| 142 EXPECT_EQ(url_fetchers_created_count(), 1); | |
| 143 } | |
| 144 | |
| 145 TEST_F(GCMNetworkChannelTest, FailedRegister) { | |
| 146 // After construction GCMNetworkChannel should have called Register. | |
| 147 EXPECT_FALSE(delegate()->register_callback.is_null()); | |
| 148 // Return error from Register call. | |
| 149 delegate()->register_callback.Run("", gcm::GCMClient::SERVER_ERROR); | |
| 150 | |
| 151 network_channel()->SendMessage("abra.cadabra"); | |
| 152 // SendMessage shouldn't trigger RequestToken. | |
| 153 EXPECT_TRUE(delegate()->request_token_callback.is_null()); | |
| 154 EXPECT_EQ(url_fetchers_created_count(), 0); | |
| 155 } | |
| 156 | |
| 157 TEST_F(GCMNetworkChannelTest, RegisterFinishesAfterSendMessage) { | |
| 158 GURL url("http://invalid.url.com"); | |
| 159 url_fetcher_factory()->SetFakeResponse(url, "", net::HTTP_OK, | |
| 160 net::URLRequestStatus::SUCCESS); | |
| 161 | |
| 162 // After construction GCMNetworkChannel should have called Register. | |
| 163 EXPECT_FALSE(delegate()->register_callback.is_null()); | |
| 164 | |
| 165 network_channel()->SendMessage("abra.cadabra"); | |
| 166 // SendMessage shouldn't trigger RequestToken. | |
| 167 EXPECT_TRUE(delegate()->request_token_callback.is_null()); | |
| 168 EXPECT_EQ(url_fetchers_created_count(), 0); | |
| 169 | |
| 170 // Return valid registration id. | |
| 171 delegate()->register_callback.Run("registration.id", gcm::GCMClient::SUCCESS); | |
| 172 | |
| 173 EXPECT_FALSE(delegate()->request_token_callback.is_null()); | |
| 174 EXPECT_EQ(url_fetchers_created_count(), 0); | |
| 175 // Return valid access token. This should trigger HTTP request. | |
| 176 delegate()->request_token_callback.Run( | |
| 177 GoogleServiceAuthError::AuthErrorNone(), "access.token"); | |
| 178 { | |
| 179 base::RunLoop run_loop; | |
| 180 run_loop.RunUntilIdle(); | |
| 181 } | |
| 182 EXPECT_EQ(url_fetchers_created_count(), 1); | |
| 183 } | |
| 184 | |
| 185 TEST_F(GCMNetworkChannelTest, RequestTokenFailure) { | |
| 186 // After construction GCMNetworkChannel should have called Register. | |
| 187 EXPECT_FALSE(delegate()->register_callback.is_null()); | |
| 188 // Return valid registration id. | |
| 189 delegate()->register_callback.Run("registration.id", gcm::GCMClient::SUCCESS); | |
| 190 | |
| 191 network_channel()->SendMessage("abra.cadabra"); | |
| 192 // SendMessage should have triggered RequestToken. No HTTP request should be | |
| 193 // started yet. | |
| 194 EXPECT_FALSE(delegate()->request_token_callback.is_null()); | |
| 195 EXPECT_EQ(url_fetchers_created_count(), 0); | |
| 196 // RequestToken returns failure. | |
| 197 delegate()->request_token_callback.Run( | |
| 198 GoogleServiceAuthError::FromConnectionError(1), ""); | |
| 199 | |
| 200 // Should be no HTTP requests. | |
| 201 EXPECT_EQ(url_fetchers_created_count(), 0); | |
| 202 } | |
| 203 | |
| 204 TEST_F(GCMNetworkChannelTest, AuthErrorFromTango) { | |
|
tim (not reviewing)
2014/01/17 19:33:42
nit - avoid the word 'Tango' in chromium.
pavely
2014/01/17 22:52:26
Done.
| |
| 205 // Setup fake response to return AUTH_ERROR. | |
| 206 GURL url("http://invalid.url.com"); | |
| 207 url_fetcher_factory()->SetFakeResponse(url, "", net::HTTP_UNAUTHORIZED, | |
| 208 net::URLRequestStatus::SUCCESS); | |
| 209 | |
| 210 // After construction GCMNetworkChannel should have called Register. | |
| 211 EXPECT_FALSE(delegate()->register_callback.is_null()); | |
| 212 // Return valid registration id. | |
| 213 delegate()->register_callback.Run("registration.id", gcm::GCMClient::SUCCESS); | |
| 214 | |
| 215 network_channel()->SendMessage("abra.cadabra"); | |
| 216 // SendMessage should have triggered RequestToken. No HTTP request should be | |
| 217 // started yet. | |
| 218 EXPECT_FALSE(delegate()->request_token_callback.is_null()); | |
| 219 EXPECT_EQ(url_fetchers_created_count(), 0); | |
| 220 // Return valid access token. This should trigger HTTP request. | |
| 221 delegate()->request_token_callback.Run( | |
| 222 GoogleServiceAuthError::AuthErrorNone(), "access.token"); | |
| 223 { | |
| 224 base::RunLoop run_loop; | |
| 225 run_loop.RunUntilIdle(); | |
| 226 } | |
| 227 EXPECT_EQ(url_fetchers_created_count(), 1); | |
| 228 EXPECT_EQ(delegate()->invalidated_token, "access.token"); | |
| 229 } | |
| 230 | |
| 231 // Following two tests are to check for memory leaks/crashes when Register and | |
| 232 // RequestToken don't complete by the teardown. | |
| 233 TEST_F(GCMNetworkChannelTest, RegisterNeverCompletes) { | |
| 234 network_channel()->SendMessage("abra.cadabra"); | |
| 235 // Register should be called by now. Let's not complete and see what happens. | |
| 236 EXPECT_FALSE(delegate()->register_callback.is_null()); | |
| 237 } | |
| 238 | |
| 239 TEST_F(GCMNetworkChannelTest, RequestTokenNeverCompletes) { | |
| 240 network_channel()->SendMessage("abra.cadabra"); | |
| 241 // Return valid registration id. | |
| 242 delegate()->register_callback.Run("registration.id", gcm::GCMClient::SUCCESS); | |
| 243 // RequestToken should be called by now. Let's not complete and see what | |
| 244 // happens. | |
| 245 EXPECT_FALSE(delegate()->request_token_callback.is_null()); | |
| 246 } | |
| 247 | |
| 39 } // namespace | 248 } // namespace |
| 40 } // namespace syncer | 249 } // namespace syncer |
| OLD | NEW |