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

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

Issue 3005049: This CL will introduce a new way to do exponential back-off on failure within... (Closed) Base URL: svn://chrome-svn/chrome/trunk/src/
Patch Set: '' 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
Property Changes:
Added: svn:eol-style
+ LF
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_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
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