Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(109)

Side by Side Diff: net/request_throttler/request_throttler_unittest.cc

Issue 2487001: Request Throttler : Exponential back-off (Closed) Base URL: http://src.chromium.org/svn/trunk/src/
Patch Set: Corrected a flaky test due to having 2 implementation of request throttling Created 10 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
(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 }
OLDNEW
« no previous file with comments | « net/request_throttler/request_throttler_manager.cc ('k') | net/url_request/url_request_http_job.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698