1 // Copyright 2013 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.
8 #include "base/logging.h"
9 #include "base/memory/scoped_ptr.h"
10 #include "base/memory/scoped_vector.h"
11 #include "base/strings/string_number_conversions.h"
12 #include "base/time/time.h"
13 #include "chrome/browser/net/evicted_domain_cookie_counter.h"
14 #include "net/cookies/canonical_cookie.h"
15 #include "net/cookies/cookie_monster.h"
16 #include "testing/gtest/include/gtest/gtest.h"
19 namespace chrome_browser_net {
22 using base::TimeDelta;
26 const char* google_url1 = "http://www.google.com";
27 const char* google_url2 = "http://mail.google.com";
28 const char* other_url1 = "http://www.example.com";
29 const char* other_url2 = "http://www.example.co.uk";
31 class EvictedDomainCookieCounterTest : public testing::Test {
33 class MockDelegate : public EvictedDomainCookieCounter::Delegate {
35 explicit MockDelegate(EvictedDomainCookieCounterTest* tester);
37 // EvictedDomainCookieCounter::Delegate implementation.
39 const EvictedDomainCookieCounter::EvictedCookie& evicted_cookie,
40 const Time& reinstatement_time) OVERRIDE;
41 virtual Time CurrentTime() const OVERRIDE;
44 EvictedDomainCookieCounterTest* tester_;
47 EvictedDomainCookieCounterTest();
48 virtual ~EvictedDomainCookieCounterTest();
50 // testing::Test implementation.
51 virtual void SetUp() OVERRIDE;
52 virtual void TearDown() OVERRIDE;
54 // Initialization that allows parameters to be specified.
55 void InitCounter(size_t max_size, size_t purge_count);
57 // Wrapper to allocate new cookie and store it in |cookies_|.
58 // If |max_age| == 0, then the cookie does not expire.
60 const char* url, const std::string& cookie_line, int64 max_age);
62 // Clears |cookies_| and creates common cookies for multiple tests.
63 void InitStockCookies();
65 // Sets simulation time to |rel_time|.
66 void GotoTime(int64 rel_time);
68 // Simulates time-passage by |delta_second|.
69 void StepTime(int64 delta_second);
71 // Simulates cookie addition or update.
72 void Add(net::CanonicalCookie* cookie);
74 // Simulates cookie removal.
75 void Remove(net::CanonicalCookie* cookie);
77 // Simulates cookie eviction.
78 void Evict(net::CanonicalCookie* cookie);
80 // For semi-realism, time considered are relative to |mock_time_base_|.
84 // To store allocated cookies for reuse.
85 ScopedVector<net::CanonicalCookie> cookies_;
87 scoped_refptr<EvictedDomainCookieCounter> cookie_counter_;
89 // Statistics as comma-separated string of duration (in seconds) between
90 // eviction and reinstatement for each cookie, in the order of eviction.
91 std::string google_stat_;
92 std::string other_stat_;
95 EvictedDomainCookieCounterTest::MockDelegate::MockDelegate(
96 EvictedDomainCookieCounterTest* tester)
99 void EvictedDomainCookieCounterTest::MockDelegate::Report(
100 const EvictedDomainCookieCounter::EvictedCookie& evicted_cookie,
101 const Time& reinstatement_time) {
102 std::string& dest = evicted_cookie.is_google ?
103 tester_->google_stat_ : tester_->other_stat_;
106 TimeDelta delta(reinstatement_time - evicted_cookie.eviction_time);
107 dest.append(base::Int64ToString(delta.InSeconds()));
110 Time EvictedDomainCookieCounterTest::MockDelegate::CurrentTime() const {
111 return tester_->mock_time_;
114 EvictedDomainCookieCounterTest::EvictedDomainCookieCounterTest() {}
116 EvictedDomainCookieCounterTest::~EvictedDomainCookieCounterTest() {}
118 void EvictedDomainCookieCounterTest::SetUp() {
119 mock_time_base_ = Time::Now() - TimeDelta::FromHours(1);
120 mock_time_ = mock_time_base_;
123 void EvictedDomainCookieCounterTest::TearDown() {
126 void EvictedDomainCookieCounterTest::InitCounter(size_t max_size,
127 size_t purge_count) {
128 scoped_ptr<MockDelegate> cookie_counter_delegate(new MockDelegate(this));
129 cookie_counter_ = new EvictedDomainCookieCounter(
131 cookie_counter_delegate.PassAs<EvictedDomainCookieCounter::Delegate>(),
136 void EvictedDomainCookieCounterTest::CreateNewCookie(
137 const char* url, const std::string& cookie_line, int64 max_age) {
138 std::string line(cookie_line);
140 line.append(";max-age=" + base::Int64ToString(max_age));
141 net::CanonicalCookie* cookie = net::CanonicalCookie::Create(
142 GURL(url), line, mock_time_, net::CookieOptions());
144 cookies_.push_back(cookie);
147 void EvictedDomainCookieCounterTest::InitStockCookies() {
149 CreateNewCookie(google_url1, "a1=1", 3000); // cookies_[0].
150 CreateNewCookie(google_url2, "a2=1", 2000); // cookies_[1].
151 CreateNewCookie(other_url1, "a1=1", 1000); // cookies_[2].
152 CreateNewCookie(other_url1, "a2=1", 1001); // cookies_[3].
153 CreateNewCookie(google_url1, "a1=1;Path=/sub", 999); // cookies_[4].
154 CreateNewCookie(other_url2, "a2=1", 0); // cookies_[5].
157 void EvictedDomainCookieCounterTest::GotoTime(int64 rel_time) {
158 mock_time_ = mock_time_base_ + TimeDelta::FromSeconds(rel_time);
161 void EvictedDomainCookieCounterTest::StepTime(int64 delta_second) {
162 mock_time_ += TimeDelta::FromSeconds(delta_second);
165 void EvictedDomainCookieCounterTest::Add(net::CanonicalCookie* cookie) {
166 cookie_counter_->OnCookieChanged(
167 *cookie, false, net::CookieMonster::Delegate::CHANGE_COOKIE_EXPLICIT);
170 void EvictedDomainCookieCounterTest::Remove(net::CanonicalCookie* cookie) {
171 cookie_counter_->OnCookieChanged(
172 *cookie, true, net::CookieMonster::Delegate::CHANGE_COOKIE_EXPLICIT);
175 void EvictedDomainCookieCounterTest::Evict(net::CanonicalCookie* cookie) {
176 cookie_counter_->OnCookieChanged(
177 *cookie, true, net::CookieMonster::Delegate::CHANGE_COOKIE_EVICTED);
180 // EvictedDomainCookieCounter takes (and owns) a CookieMonster::Delegate for
181 // chaining. To ensure that the chaining indeed occurs, we implement a
182 // dummy CookieMonster::Delegate to increment an integer.
183 TEST_F(EvictedDomainCookieCounterTest, TestChain) {
186 class ChangedDelegateDummy : public net::CookieMonster::Delegate {
188 explicit ChangedDelegateDummy(int* result) : result_(result) {}
190 virtual void OnCookieChanged(const net::CanonicalCookie& cookie,
192 ChangeCause cause) OVERRIDE {
196 virtual void OnLoaded() OVERRIDE {}
199 virtual ~ChangedDelegateDummy() {}
204 scoped_ptr<MockDelegate> cookie_counter_delegate(new MockDelegate(this));
205 cookie_counter_ = new EvictedDomainCookieCounter(
206 new ChangedDelegateDummy(&result),
207 cookie_counter_delegate.PassAs<EvictedDomainCookieCounter::Delegate>(),
211 // Perform 6 cookie transactions.
212 for (int i = 0; i < 6; ++i) {
219 EXPECT_EQ(18, result); // 6 cookies x 3 operations each.
222 // Basic flow: add cookies, evict, then reinstate.
223 TEST_F(EvictedDomainCookieCounterTest, TestBasicFlow) {
226 // Add all cookies at (relative time) t = 0.
227 for (int i = 0; i < 6; ++i)
229 EXPECT_EQ(0u, cookie_counter_->GetStorageSize()); // No activities on add.
230 EXPECT_EQ(";", google_stat_ + ";" + other_stat_);
231 // Evict cookies at t = [1,3,6,10,15,21].
232 for (int i = 0; i < 6; ++i) {
236 EXPECT_EQ(6u, cookie_counter_->GetStorageSize()); // Storing all evictions.
237 EXPECT_EQ(";", google_stat_ + ";" + other_stat_);
238 // Reinstate cookies at t = [22,23,24,25,26,27].
239 for (int i = 0; i < 6; ++i) {
243 EXPECT_EQ(0u, cookie_counter_->GetStorageSize()); // Everything is removed.
244 // Expected reinstatement delays: [21,20,18,15,11,6].
245 EXPECT_EQ("21,20,11;18,15,6", google_stat_ + ";" + other_stat_);
248 // Removed cookies are ignored by EvictedDomainCookieCounter.
249 TEST_F(EvictedDomainCookieCounterTest, TestRemove) {
252 // Add all cookies at (relative time) t = 0.
253 for (int i = 0; i < 6; ++i)
255 // Remove cookies at t = [1,3,6,10,15,21].
256 for (int i = 0; i < 6; ++i) {
260 EXPECT_EQ(0u, cookie_counter_->GetStorageSize());
261 // Add cookies again at t = [22,23,24,25,26,27].
262 for (int i = 0; i < 5; ++i) {
266 EXPECT_EQ(0u, cookie_counter_->GetStorageSize());
267 // No cookies were evicted, so no reinstatement take place.
268 EXPECT_EQ(";", google_stat_ + ";" + other_stat_);
271 // Expired cookies should not be counted by EvictedDomainCookieCounter.
272 TEST_F(EvictedDomainCookieCounterTest, TestExpired) {
275 // Add all cookies at (relative time) t = 0.
276 for (int i = 0; i < 6; ++i)
278 // Evict cookies at t = [1,3,6,10,15,21].
279 for (int i = 0; i < 6; ++i) {
283 EXPECT_EQ(6u, cookie_counter_->GetStorageSize());
284 GotoTime(1000); // t = 1000, so cookies_[2,4] expire.
286 // Reinstate cookies at t = [1000,1000,(1000),1000,(1000),1000].
287 InitStockCookies(); // Refresh cookies, so new cookies expire in the future.
288 for (int i = 0; i < 6; ++i)
290 EXPECT_EQ(0u, cookie_counter_->GetStorageSize());
291 // Reinstatement delays: [999,997,(994),990,(985),979].
292 EXPECT_EQ("999,997;990,979", google_stat_ + ";" + other_stat_);
295 // Garbage collection should remove the oldest evicted cookies.
296 TEST_F(EvictedDomainCookieCounterTest, TestGarbageCollection) {
297 InitCounter(4, 2); // Reduced capacity.
299 // Add all cookies at (relative time) t = 0.
300 for (int i = 0; i < 6; ++i)
302 // Evict cookies at t = [1,3,6,10].
303 for (int i = 0; i < 4; ++i) {
307 EXPECT_EQ(4u, cookie_counter_->GetStorageSize()); // Reached capacity.
309 Evict(cookies_[4]); // Evict at t = 15, garbage collection takes place.
310 EXPECT_EQ(2u, cookie_counter_->GetStorageSize());
312 Evict(cookies_[5]); // Evict at t = 21.
313 EXPECT_EQ(3u, cookie_counter_->GetStorageSize());
314 EXPECT_EQ(";", google_stat_ + ";" + other_stat_);
315 // Reinstate cookies at t = [(100),(100),(100),100,100,100].
317 for (int i = 0; i < 6; ++i)
319 // Expected reinstatement delays: [(99),(97),(94),90,85,79]
320 EXPECT_EQ("85;90,79", google_stat_ + ";" + other_stat_);
323 // Garbage collection should remove the specified number of evicted cookies
324 // even when there are ties amongst oldest evicted cookies.
325 TEST_F(EvictedDomainCookieCounterTest, TestGarbageCollectionTie) {
327 // Add 10 cookies at time [0,1,3,6,...,45]
328 for (int i = 0; i < 10; ++i) {
330 CreateNewCookie(google_url1, "a" + base::IntToString(i) + "=1", 3000);
333 // Evict 6 cookies at t = [100,...,100].
335 for (int i = 0; i < 6; ++i)
337 EXPECT_EQ(6u, cookie_counter_->GetStorageSize());
338 // Evict 3 cookies at t = [210,220,230].
340 for (int i = 6; i < 9; ++i) {
344 EXPECT_EQ(9u, cookie_counter_->GetStorageSize()); // Reached capacity.
345 // Evict 1 cookie at t = 300, and garbage collection takes place.
348 // Some arbitrary 4 out of 6 cookies evicted at t = 100 are gone from storage.
349 EXPECT_EQ(6u, cookie_counter_->GetStorageSize()); // 10 - 4.
350 // Reinstate cookies at t = [400,...,400].
352 for (int i = 0; i < 10; ++i)
354 EXPECT_EQ(0u, cookie_counter_->GetStorageSize());
355 // Expected reinstatement delays:
356 // [300,300,300,300,300,300 <= keeping 2 only,190,180,170,100].
357 EXPECT_EQ("300,300,190,180,170,100;", google_stat_ + ";" + other_stat_);
360 // Garbage collection prioritize removal of expired cookies.
361 TEST_F(EvictedDomainCookieCounterTest, TestGarbageCollectionWithExpiry) {
364 // Add all cookies at (relative time) t = 0.
365 for (int i = 0; i < 6; ++i)
367 // Evict cookies at t = [1,3,6,10,15].
368 for (int i = 0; i < 5; ++i) {
372 EXPECT_EQ(5u, cookie_counter_->GetStorageSize()); // Reached capacity.
373 GotoTime(1200); // t = 1200, so cookies_[2,3,4] expire.
374 // Evict cookies_[5] (not expired) at t = 1200.
376 // Garbage collection would have taken place, removing 3 expired cookies,
377 // so that there's no need to remove more.
378 EXPECT_EQ(3u, cookie_counter_->GetStorageSize());
379 // Reinstate cookies at t = [1500,1500,(1500),(1500),(1500),1500].
381 InitStockCookies(); // Refresh cookies, so new cookies expire in the future.
382 for (int i = 0; i < 6; ++i)
384 EXPECT_EQ(0u, cookie_counter_->GetStorageSize());
385 // Reinstatement delays: [1499,1497,(1494),(1490),(1485),300].
386 EXPECT_EQ("1499,1497;300", google_stat_ + ";" + other_stat_);
391 } // namespace chrome_browser_net