| OLD | NEW |
| (Empty) | |
| 1 // Copyright (c) 2010 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 |
| 5 #include "base/pickle.h" |
| 6 #include "base/scoped_ptr.h" |
| 7 #include "base/string_util.h" |
| 8 #include "base/time.h" |
| 9 #include "net/base/test_completion_callback.h" |
| 10 #include "net/request_throttler/request_throttler_manager.h" |
| 11 #include "net/request_throttler/request_throttler_header_interface.h" |
| 12 #include "net/url_request/url_request_context.h" |
| 13 #include "testing/gtest/include/gtest/gtest.h" |
| 14 |
| 15 using base::TimeDelta; |
| 16 using base::TimeTicks; |
| 17 |
| 18 namespace { |
| 19 class MockRequestThrottlerManager; |
| 20 |
| 21 class MockRequestThrottlerEntry : public RequestThrottlerEntry { |
| 22 public : |
| 23 MockRequestThrottlerEntry() {} |
| 24 MockRequestThrottlerEntry(TimeTicks release_time, TimeTicks fake_now) |
| 25 : fake_time_now_(fake_now) { |
| 26 release_time_ = release_time; |
| 27 } |
| 28 virtual ~MockRequestThrottlerEntry() {} |
| 29 |
| 30 void ResetToBlank(TimeTicks time_now) { |
| 31 fake_time_now_ = time_now; |
| 32 release_time_ = time_now; |
| 33 num_times_delayed_ = 0; |
| 34 } |
| 35 |
| 36 // Overloaded for test |
| 37 virtual TimeTicks GetTimeNow() const { return fake_time_now_; } |
| 38 |
| 39 TimeTicks release_time() const { return release_time_; } |
| 40 void set_release_time(TimeTicks time) { release_time_ = time; } |
| 41 |
| 42 TimeTicks fake_time_now_; |
| 43 }; |
| 44 |
| 45 class MockRequestThrottlerHeaderAdapter |
| 46 : public RequestThrottlerHeaderInterface { |
| 47 public: |
| 48 MockRequestThrottlerHeaderAdapter() |
| 49 : fake_retry_value_("0.0"), |
| 50 fake_response_code_(0) { |
| 51 } |
| 52 |
| 53 MockRequestThrottlerHeaderAdapter(const std::string& retry_value, |
| 54 const int response_code) |
| 55 : fake_retry_value_(retry_value), |
| 56 fake_response_code_(response_code) { |
| 57 } |
| 58 |
| 59 ~MockRequestThrottlerHeaderAdapter() {} |
| 60 |
| 61 virtual std::string GetNormalizedValue(const std::string& key) const { |
| 62 if (key == MockRequestThrottlerEntry::kRetryHeaderName) |
| 63 return fake_retry_value_; |
| 64 return ""; |
| 65 } |
| 66 |
| 67 virtual int GetResponseCode() const { return fake_response_code_; } |
| 68 |
| 69 std::string fake_retry_value_; |
| 70 int fake_response_code_; |
| 71 }; |
| 72 |
| 73 class MockRequestThrottlerManager : public RequestThrottlerManager { |
| 74 public: |
| 75 MockRequestThrottlerManager() {} |
| 76 virtual ~MockRequestThrottlerManager() {} |
| 77 |
| 78 // Method to process the url using RequestThrottlerManager protected method. |
| 79 std::string DoGetUrlIdFromUrl(const GURL& url) { return GetIdFromUrl(url); } |
| 80 |
| 81 // Method to use the garbage collecting method of RequestThrottlerManager. |
| 82 void DoGarbageCollectEntries() { GarbageCollectEntries(); } |
| 83 |
| 84 // Returns the number of entries in the host map. |
| 85 int GetNumberOfEntries() { return url_entries_.size(); } |
| 86 |
| 87 void CreateEntry(const bool is_outdated); |
| 88 |
| 89 static int create_entry_index; |
| 90 }; |
| 91 |
| 92 /*------------------ Mock request throttler manager --------------------------*/ |
| 93 int MockRequestThrottlerManager::create_entry_index = 0; |
| 94 |
| 95 void MockRequestThrottlerManager::CreateEntry(const bool is_outdated) { |
| 96 TimeTicks time = TimeTicks::Now(); |
| 97 if (is_outdated) { |
| 98 time -= TimeDelta::FromSeconds( |
| 99 MockRequestThrottlerEntry::kEntryLifetimeSec); |
| 100 time -= TimeDelta::FromSeconds(1); |
| 101 } |
| 102 std::string index = IntToString(create_entry_index++); |
| 103 url_entries_[index] = new MockRequestThrottlerEntry(time, TimeTicks::Now()); |
| 104 } |
| 105 |
| 106 /* ---------------- Request throttler entry test -----------------------------*/ |
| 107 struct TimeAndBool { |
| 108 TimeAndBool(TimeTicks time_value, bool expected, int line_num) { |
| 109 time = time_value; |
| 110 result = expected; |
| 111 line = line_num; |
| 112 } |
| 113 TimeTicks time; |
| 114 bool result; |
| 115 int line; |
| 116 }; |
| 117 |
| 118 struct GurlAndString { |
| 119 GurlAndString(GURL url_value, const std::string& expected, int line_num) { |
| 120 url = url_value; |
| 121 result = expected; |
| 122 line = line_num; |
| 123 } |
| 124 GURL url; |
| 125 std::string result; |
| 126 int line; |
| 127 }; |
| 128 |
| 129 } // namespace |
| 130 |
| 131 class RequestThrottlerEntryTest : public ::testing::Test { |
| 132 protected: |
| 133 virtual void SetUp(); |
| 134 TimeTicks now_; |
| 135 TimeDelta delay_; |
| 136 scoped_refptr<MockRequestThrottlerEntry> entry_; |
| 137 }; |
| 138 |
| 139 void RequestThrottlerEntryTest::SetUp() { |
| 140 now_ = TimeTicks::Now(); |
| 141 entry_ = new MockRequestThrottlerEntry(); |
| 142 entry_->ResetToBlank(now_); |
| 143 } |
| 144 |
| 145 std::ostream& operator<<(std::ostream& out, const base::TimeTicks& time) { |
| 146 return out << time.ToInternalValue(); |
| 147 } |
| 148 |
| 149 TEST_F(RequestThrottlerEntryTest, InterfaceRequestNotAllowed) { |
| 150 entry_->set_release_time(entry_->fake_time_now_ + |
| 151 TimeDelta::FromMilliseconds(1)); |
| 152 EXPECT_FALSE(entry_->IsRequestAllowed()); |
| 153 } |
| 154 |
| 155 TEST_F(RequestThrottlerEntryTest, InterfaceRequestAllowed) { |
| 156 entry_->set_release_time(entry_->fake_time_now_); |
| 157 EXPECT_TRUE(entry_->IsRequestAllowed()); |
| 158 entry_->set_release_time(entry_->fake_time_now_ - |
| 159 TimeDelta::FromMilliseconds(1)); |
| 160 EXPECT_TRUE(entry_->IsRequestAllowed()); |
| 161 } |
| 162 |
| 163 TEST_F(RequestThrottlerEntryTest, InterfaceUpdateRetryAfter) { |
| 164 // If the response we received as a retry-after field, |
| 165 // the request should be delayed. |
| 166 MockRequestThrottlerHeaderAdapter header_w_delay_header("5.5", 200); |
| 167 entry_->UpdateWithResponse(&header_w_delay_header); |
| 168 EXPECT_GT(entry_->release_time(), entry_->fake_time_now_) << |
| 169 "When the server put a positive value in retry-after we should " |
| 170 "increase release_time"; |
| 171 |
| 172 entry_->ResetToBlank(now_); |
| 173 header_w_delay_header.fake_retry_value_ = "-5.5"; |
| 174 EXPECT_EQ(entry_->release_time(), entry_->fake_time_now_) << |
| 175 "When given a negative value, it should not change the release_time"; |
| 176 } |
| 177 |
| 178 TEST_F(RequestThrottlerEntryTest, InterfaceUpdateFailure) { |
| 179 MockRequestThrottlerHeaderAdapter failure_response("0", 505); |
| 180 entry_->UpdateWithResponse(&failure_response); |
| 181 EXPECT_GT(entry_->release_time(), entry_->fake_time_now_) << |
| 182 "A failure should increase the release_time"; |
| 183 } |
| 184 |
| 185 TEST_F(RequestThrottlerEntryTest, InterfaceUpdateSuccess) { |
| 186 MockRequestThrottlerHeaderAdapter success_response("0", 200); |
| 187 entry_->UpdateWithResponse(&success_response); |
| 188 EXPECT_EQ(entry_->release_time(), entry_->fake_time_now_) << |
| 189 "A success should not add any delay"; |
| 190 } |
| 191 |
| 192 TEST_F(RequestThrottlerEntryTest, InterfaceUpdateSuccessThenFailure) { |
| 193 MockRequestThrottlerHeaderAdapter failure_response("0", 500), |
| 194 success_response("0", 200); |
| 195 entry_->UpdateWithResponse(&success_response); |
| 196 entry_->UpdateWithResponse(&failure_response); |
| 197 EXPECT_GT(entry_->release_time(), entry_->fake_time_now_) << |
| 198 "This scenario should add delay"; |
| 199 } |
| 200 |
| 201 TEST_F(RequestThrottlerEntryTest, IsEntryReallyOutdated) { |
| 202 TimeDelta lifetime = TimeDelta::FromSeconds( |
| 203 MockRequestThrottlerEntry::kEntryLifetimeSec); |
| 204 const TimeDelta kFiveMs = TimeDelta::FromMilliseconds(5); |
| 205 |
| 206 TimeAndBool test_values[] = { |
| 207 TimeAndBool(now_, false, __LINE__), |
| 208 TimeAndBool(now_ - kFiveMs, false, __LINE__), |
| 209 TimeAndBool(now_ + kFiveMs, false, __LINE__), |
| 210 TimeAndBool(now_ - lifetime, false, __LINE__), |
| 211 TimeAndBool(now_ - (lifetime + kFiveMs), true, __LINE__)}; |
| 212 |
| 213 for (unsigned int i = 0; i < arraysize(test_values); ++i) { |
| 214 entry_->set_release_time(test_values[i].time); |
| 215 EXPECT_EQ(entry_->IsEntryOutdated(), test_values[i].result) << |
| 216 "Test case #" << i << " line " << test_values[i].line << " failed"; |
| 217 } |
| 218 } |
| 219 |
| 220 TEST_F(RequestThrottlerEntryTest, MaxAllowedBackoff) { |
| 221 for (int i = 0; i < 30; ++i) { |
| 222 MockRequestThrottlerHeaderAdapter response_adapter("0.0", 505); |
| 223 entry_->UpdateWithResponse(&response_adapter); |
| 224 } |
| 225 |
| 226 delay_ = entry_->release_time() - now_; |
| 227 EXPECT_EQ(delay_.InMilliseconds(), |
| 228 MockRequestThrottlerEntry::kMaximumBackoffMs); |
| 229 } |
| 230 |
| 231 TEST_F(RequestThrottlerEntryTest, MalformedContent) { |
| 232 for (int i = 0; i < 5; ++i) { |
| 233 MockRequestThrottlerHeaderAdapter response_adapter("0.0", 505); |
| 234 entry_->UpdateWithResponse(&response_adapter); |
| 235 } |
| 236 TimeTicks release_after_failures = entry_->release_time(); |
| 237 |
| 238 // Send a success code to reset backoff. |
| 239 MockRequestThrottlerHeaderAdapter response_adapter("0.0", 200); |
| 240 entry_->UpdateWithResponse(&response_adapter); |
| 241 EXPECT_EQ(entry_->release_time(), release_after_failures); |
| 242 |
| 243 // Then inform the entry that previous package was malformed |
| 244 // it is suppose to regenerate previous state. |
| 245 entry_->ReceivedContentWasMalformed(); |
| 246 EXPECT_GT(entry_->release_time(), release_after_failures); |
| 247 } |
| 248 |
| 249 TEST(RequestThrottlerManager, IsUrlStandardised) { |
| 250 MockRequestThrottlerManager manager; |
| 251 GurlAndString test_values[] = { |
| 252 GurlAndString(GURL("http://www.example.com"), |
| 253 std::string("http://www.example.com/"), __LINE__), |
| 254 GurlAndString(GURL("http://www.Example.com"), |
| 255 std::string("http://www.example.com/"), __LINE__), |
| 256 GurlAndString(GURL("http://www.ex4mple.com/Pr4c71c41"), |
| 257 std::string("http://www.ex4mple.com/pr4c71c41"), __LINE__), |
| 258 GurlAndString(GURL("http://www.example.com/0/token/false"), |
| 259 std::string("http://www.example.com/0/token/false"), |
| 260 __LINE__), |
| 261 GurlAndString(GURL("http://www.example.com/index.php?code=javascript"), |
| 262 std::string("http://www.example.com/index.php"), __LINE__), |
| 263 GurlAndString(GURL("http://www.example.com/index.php?code=1#superEntry"), |
| 264 std::string("http://www.example.com/index.php"), |
| 265 __LINE__)}; |
| 266 |
| 267 for (unsigned int i = 0; i < arraysize(test_values); ++i) { |
| 268 std::string temp = manager.DoGetUrlIdFromUrl(test_values[i].url); |
| 269 EXPECT_EQ(temp, test_values[i].result) << |
| 270 "Test case #" << i << " line " << test_values[i].line << " failed"; |
| 271 } |
| 272 } |
| 273 |
| 274 TEST(RequestThrottlerManager, AreEntriesBeingCollected ) { |
| 275 MockRequestThrottlerManager manager; |
| 276 |
| 277 manager.CreateEntry(true); // true = Entry is outdated. |
| 278 manager.CreateEntry(true); |
| 279 manager.CreateEntry(true); |
| 280 manager.DoGarbageCollectEntries(); |
| 281 EXPECT_EQ(0, manager.GetNumberOfEntries()); |
| 282 |
| 283 manager.CreateEntry(false); |
| 284 manager.CreateEntry(false); |
| 285 manager.CreateEntry(false); |
| 286 manager.CreateEntry(true); |
| 287 manager.DoGarbageCollectEntries(); |
| 288 EXPECT_EQ(3, manager.GetNumberOfEntries()); |
| 289 } |
| 290 |
| 291 TEST(RequestThrottlerManager, IsHostBeingRegistered) { |
| 292 MockRequestThrottlerManager manager; |
| 293 |
| 294 manager.RegisterRequestUrl(GURL("http://www.example.com/")); |
| 295 manager.RegisterRequestUrl(GURL("http://www.google.com/")); |
| 296 manager.RegisterRequestUrl(GURL("http://www.google.com/index/0")); |
| 297 manager.RegisterRequestUrl(GURL("http://www.google.com/index/0?code=1")); |
| 298 manager.RegisterRequestUrl(GURL("http://www.google.com/index/0#lolsaure")); |
| 299 |
| 300 EXPECT_EQ(3, manager.GetNumberOfEntries()); |
| 301 } |
| OLD | NEW |