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