|  | // Copyright 2013 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "base/file_util.h" | 
|  | #include "base/message_loop/message_loop_proxy.h" | 
|  | #include "base/run_loop.h" | 
|  | #include "base/threading/thread.h" | 
|  | #include "content/browser/browser_thread_impl.h" | 
|  | #include "content/browser/gpu/shader_disk_cache.h" | 
|  | #include "content/browser/quota/mock_quota_manager.h" | 
|  | #include "content/browser/storage_partition_impl.h" | 
|  | #include "content/public/browser/local_storage_usage_info.h" | 
|  | #include "content/public/browser/storage_partition.h" | 
|  | #include "content/public/test/mock_special_storage_policy.h" | 
|  | #include "content/public/test/test_browser_context.h" | 
|  | #include "content/public/test/test_browser_thread.h" | 
|  | #include "content/public/test/test_browser_thread_bundle.h" | 
|  | #include "net/base/test_completion_callback.h" | 
|  | #include "net/cookies/cookie_monster.h" | 
|  | #include "net/url_request/url_request_context.h" | 
|  | #include "net/url_request/url_request_context_getter.h" | 
|  | #include "testing/gtest/include/gtest/gtest.h" | 
|  | #include "webkit/browser/quota/quota_manager.h" | 
|  |  | 
|  | namespace content { | 
|  | namespace { | 
|  |  | 
|  | const int kDefaultClientId = 42; | 
|  | const char kCacheKey[] = "key"; | 
|  | const char kCacheValue[] = "cached value"; | 
|  |  | 
|  | const char kTestOrigin1[] = "http://host1:1/"; | 
|  | const char kTestOrigin2[] = "http://host2:1/"; | 
|  | const char kTestOrigin3[] = "http://host3:1/"; | 
|  | const char kTestOriginDevTools[] = "chrome-devtools://abcdefghijklmnopqrstuvw/"; | 
|  |  | 
|  | const GURL kOrigin1(kTestOrigin1); | 
|  | const GURL kOrigin2(kTestOrigin2); | 
|  | const GURL kOrigin3(kTestOrigin3); | 
|  | const GURL kOriginDevTools(kTestOriginDevTools); | 
|  |  | 
|  | const base::FilePath::CharType kDomStorageOrigin1[] = | 
|  | FILE_PATH_LITERAL("http_host1_1.localstorage"); | 
|  |  | 
|  | const base::FilePath::CharType kDomStorageOrigin2[] = | 
|  | FILE_PATH_LITERAL("http_host2_1.localstorage"); | 
|  |  | 
|  | const base::FilePath::CharType kDomStorageOrigin3[] = | 
|  | FILE_PATH_LITERAL("http_host3_1.localstorage"); | 
|  |  | 
|  | const storage::StorageType kTemporary = storage::kStorageTypeTemporary; | 
|  | const storage::StorageType kPersistent = storage::kStorageTypePersistent; | 
|  |  | 
|  | const storage::QuotaClient::ID kClientFile = storage::QuotaClient::kFileSystem; | 
|  |  | 
|  | const uint32 kAllQuotaRemoveMask = | 
|  | StoragePartition::REMOVE_DATA_MASK_APPCACHE | | 
|  | StoragePartition::REMOVE_DATA_MASK_FILE_SYSTEMS | | 
|  | StoragePartition::REMOVE_DATA_MASK_INDEXEDDB | | 
|  | StoragePartition::REMOVE_DATA_MASK_WEBSQL; | 
|  |  | 
|  | class AwaitCompletionHelper { | 
|  | public: | 
|  | AwaitCompletionHelper() : start_(false), already_quit_(false) {} | 
|  | virtual ~AwaitCompletionHelper() {} | 
|  |  | 
|  | void BlockUntilNotified() { | 
|  | if (!already_quit_) { | 
|  | DCHECK(!start_); | 
|  | start_ = true; | 
|  | base::MessageLoop::current()->Run(); | 
|  | } else { | 
|  | DCHECK(!start_); | 
|  | already_quit_ = false; | 
|  | } | 
|  | } | 
|  |  | 
|  | void Notify() { | 
|  | if (start_) { | 
|  | DCHECK(!already_quit_); | 
|  | base::MessageLoop::current()->Quit(); | 
|  | start_ = false; | 
|  | } else { | 
|  | DCHECK(!already_quit_); | 
|  | already_quit_ = true; | 
|  | } | 
|  | } | 
|  |  | 
|  | private: | 
|  | // Helps prevent from running message_loop, if the callback invoked | 
|  | // immediately. | 
|  | bool start_; | 
|  | bool already_quit_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(AwaitCompletionHelper); | 
|  | }; | 
|  |  | 
|  | class RemoveCookieTester { | 
|  | public: | 
|  | explicit RemoveCookieTester(TestBrowserContext* context) | 
|  | : get_cookie_success_(false), monster_(NULL) { | 
|  | SetMonster(context->GetRequestContext()->GetURLRequestContext()-> | 
|  | cookie_store()->GetCookieMonster()); | 
|  | } | 
|  |  | 
|  | // Returns true, if the given cookie exists in the cookie store. | 
|  | bool ContainsCookie() { | 
|  | get_cookie_success_ = false; | 
|  | monster_->GetCookiesWithOptionsAsync( | 
|  | kOrigin1, net::CookieOptions(), | 
|  | base::Bind(&RemoveCookieTester::GetCookieCallback, | 
|  | base::Unretained(this))); | 
|  | await_completion_.BlockUntilNotified(); | 
|  | return get_cookie_success_; | 
|  | } | 
|  |  | 
|  | void AddCookie() { | 
|  | monster_->SetCookieWithOptionsAsync( | 
|  | kOrigin1, "A=1", net::CookieOptions(), | 
|  | base::Bind(&RemoveCookieTester::SetCookieCallback, | 
|  | base::Unretained(this))); | 
|  | await_completion_.BlockUntilNotified(); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | void SetMonster(net::CookieStore* monster) { | 
|  | monster_ = monster; | 
|  | } | 
|  |  | 
|  | private: | 
|  | void GetCookieCallback(const std::string& cookies) { | 
|  | if (cookies == "A=1") { | 
|  | get_cookie_success_ = true; | 
|  | } else { | 
|  | EXPECT_EQ("", cookies); | 
|  | get_cookie_success_ = false; | 
|  | } | 
|  | await_completion_.Notify(); | 
|  | } | 
|  |  | 
|  | void SetCookieCallback(bool result) { | 
|  | ASSERT_TRUE(result); | 
|  | await_completion_.Notify(); | 
|  | } | 
|  |  | 
|  | bool get_cookie_success_; | 
|  | AwaitCompletionHelper await_completion_; | 
|  | net::CookieStore* monster_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(RemoveCookieTester); | 
|  | }; | 
|  |  | 
|  | class RemoveLocalStorageTester { | 
|  | public: | 
|  | explicit RemoveLocalStorageTester(TestBrowserContext* profile) | 
|  | : profile_(profile), dom_storage_context_(NULL) { | 
|  | dom_storage_context_ = | 
|  | content::BrowserContext::GetDefaultStoragePartition(profile)-> | 
|  | GetDOMStorageContext(); | 
|  | } | 
|  |  | 
|  | // Returns true, if the given origin URL exists. | 
|  | bool DOMStorageExistsForOrigin(const GURL& origin) { | 
|  | GetLocalStorageUsage(); | 
|  | await_completion_.BlockUntilNotified(); | 
|  | for (size_t i = 0; i < infos_.size(); ++i) { | 
|  | if (origin == infos_[i].origin) | 
|  | return true; | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | void AddDOMStorageTestData() { | 
|  | // Note: This test depends on details of how the dom_storage library | 
|  | // stores data in the host file system. | 
|  | base::FilePath storage_path = | 
|  | profile_->GetPath().AppendASCII("Local Storage"); | 
|  | base::CreateDirectory(storage_path); | 
|  |  | 
|  | // Write some files. | 
|  | base::WriteFile(storage_path.Append(kDomStorageOrigin1), NULL, 0); | 
|  | base::WriteFile(storage_path.Append(kDomStorageOrigin2), NULL, 0); | 
|  | base::WriteFile(storage_path.Append(kDomStorageOrigin3), NULL, 0); | 
|  |  | 
|  | // Tweak their dates. | 
|  | base::Time now = base::Time::Now(); | 
|  | base::TouchFile(storage_path.Append(kDomStorageOrigin1), now, now); | 
|  |  | 
|  | base::Time one_day_ago = now - base::TimeDelta::FromDays(1); | 
|  | base::TouchFile(storage_path.Append(kDomStorageOrigin2), | 
|  | one_day_ago, one_day_ago); | 
|  |  | 
|  | base::Time sixty_days_ago = now - base::TimeDelta::FromDays(60); | 
|  | base::TouchFile(storage_path.Append(kDomStorageOrigin3), | 
|  | sixty_days_ago, sixty_days_ago); | 
|  | } | 
|  |  | 
|  | private: | 
|  | void GetLocalStorageUsage() { | 
|  | dom_storage_context_->GetLocalStorageUsage( | 
|  | base::Bind(&RemoveLocalStorageTester::OnGotLocalStorageUsage, | 
|  | base::Unretained(this))); | 
|  | } | 
|  | void OnGotLocalStorageUsage( | 
|  | const std::vector<content::LocalStorageUsageInfo>& infos) { | 
|  | infos_ = infos; | 
|  | await_completion_.Notify(); | 
|  | } | 
|  |  | 
|  | // We don't own these pointers. | 
|  | TestBrowserContext* profile_; | 
|  | content::DOMStorageContext* dom_storage_context_; | 
|  |  | 
|  | std::vector<content::LocalStorageUsageInfo> infos_; | 
|  |  | 
|  | AwaitCompletionHelper await_completion_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(RemoveLocalStorageTester); | 
|  | }; | 
|  |  | 
|  | bool IsWebSafeSchemeForTest(const std::string& scheme) { | 
|  | return scheme == "http"; | 
|  | } | 
|  |  | 
|  | bool DoesOriginMatchForUnprotectedWeb( | 
|  | const GURL& origin, | 
|  | storage::SpecialStoragePolicy* special_storage_policy) { | 
|  | if (IsWebSafeSchemeForTest(origin.scheme())) | 
|  | return !special_storage_policy->IsStorageProtected(origin.GetOrigin()); | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool DoesOriginMatchForBothProtectedAndUnprotectedWeb( | 
|  | const GURL& origin, | 
|  | storage::SpecialStoragePolicy* special_storage_policy) { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool DoesOriginMatchUnprotected( | 
|  | const GURL& origin, | 
|  | storage::SpecialStoragePolicy* special_storage_policy) { | 
|  | return origin.GetOrigin().scheme() != kOriginDevTools.scheme(); | 
|  | } | 
|  |  | 
|  | void ClearQuotaData(content::StoragePartition* partition, | 
|  | base::RunLoop* loop_to_quit) { | 
|  | partition->ClearData(kAllQuotaRemoveMask, | 
|  | StoragePartition::QUOTA_MANAGED_STORAGE_MASK_ALL, GURL(), | 
|  | StoragePartition::OriginMatcherFunction(), base::Time(), | 
|  | base::Time::Max(), loop_to_quit->QuitClosure()); | 
|  | } | 
|  |  | 
|  | void ClearQuotaDataWithOriginMatcher( | 
|  | content::StoragePartition* partition, | 
|  | const GURL& remove_origin, | 
|  | const StoragePartition::OriginMatcherFunction& origin_matcher, | 
|  | const base::Time delete_begin, | 
|  | base::RunLoop* loop_to_quit) { | 
|  | partition->ClearData(kAllQuotaRemoveMask, | 
|  | StoragePartition::QUOTA_MANAGED_STORAGE_MASK_ALL, | 
|  | remove_origin, origin_matcher, delete_begin, | 
|  | base::Time::Max(), loop_to_quit->QuitClosure()); | 
|  | } | 
|  |  | 
|  | void ClearQuotaDataForOrigin( | 
|  | content::StoragePartition* partition, | 
|  | const GURL& remove_origin, | 
|  | const base::Time delete_begin, | 
|  | base::RunLoop* loop_to_quit) { | 
|  | ClearQuotaDataWithOriginMatcher( | 
|  | partition, remove_origin, | 
|  | StoragePartition::OriginMatcherFunction(), delete_begin, | 
|  | loop_to_quit); | 
|  | } | 
|  |  | 
|  | void ClearQuotaDataForNonPersistent( | 
|  | content::StoragePartition* partition, | 
|  | const base::Time delete_begin, | 
|  | base::RunLoop* loop_to_quit) { | 
|  | partition->ClearData( | 
|  | kAllQuotaRemoveMask, | 
|  | ~StoragePartition::QUOTA_MANAGED_STORAGE_MASK_PERSISTENT, | 
|  | GURL(), StoragePartition::OriginMatcherFunction(), delete_begin, | 
|  | base::Time::Max(), loop_to_quit->QuitClosure()); | 
|  | } | 
|  |  | 
|  | void ClearCookies(content::StoragePartition* partition, | 
|  | const base::Time delete_begin, | 
|  | const base::Time delete_end, | 
|  | base::RunLoop* run_loop) { | 
|  | partition->ClearData( | 
|  | StoragePartition::REMOVE_DATA_MASK_COOKIES, | 
|  | StoragePartition::QUOTA_MANAGED_STORAGE_MASK_ALL, | 
|  | GURL(), StoragePartition::OriginMatcherFunction(), | 
|  | delete_begin, delete_end, run_loop->QuitClosure()); | 
|  | } | 
|  |  | 
|  | void ClearStuff(uint32 remove_mask, | 
|  | content::StoragePartition* partition, | 
|  | const base::Time delete_begin, | 
|  | const base::Time delete_end, | 
|  | const StoragePartition::OriginMatcherFunction& origin_matcher, | 
|  | base::RunLoop* run_loop) { | 
|  | partition->ClearData( | 
|  | remove_mask, StoragePartition::QUOTA_MANAGED_STORAGE_MASK_ALL, | 
|  | GURL(), origin_matcher, delete_begin, delete_end, | 
|  | run_loop->QuitClosure()); | 
|  | } | 
|  |  | 
|  | void ClearData(content::StoragePartition* partition, | 
|  | base::RunLoop* run_loop) { | 
|  | base::Time time; | 
|  | partition->ClearData( | 
|  | StoragePartition::REMOVE_DATA_MASK_SHADER_CACHE, | 
|  | StoragePartition::QUOTA_MANAGED_STORAGE_MASK_ALL, | 
|  | GURL(), StoragePartition::OriginMatcherFunction(), | 
|  | time, time, run_loop->QuitClosure()); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | class StoragePartitionImplTest : public testing::Test { | 
|  | public: | 
|  | StoragePartitionImplTest() | 
|  | : thread_bundle_(content::TestBrowserThreadBundle::IO_MAINLOOP), | 
|  | browser_context_(new TestBrowserContext()) { | 
|  | } | 
|  |  | 
|  | MockQuotaManager* GetMockManager() { | 
|  | if (!quota_manager_.get()) { | 
|  | quota_manager_ = new MockQuotaManager( | 
|  | browser_context_->IsOffTheRecord(), | 
|  | browser_context_->GetPath(), | 
|  | BrowserThread::GetMessageLoopProxyForThread(BrowserThread::IO).get(), | 
|  | BrowserThread::GetMessageLoopProxyForThread(BrowserThread::DB).get(), | 
|  | browser_context_->GetSpecialStoragePolicy()); | 
|  | } | 
|  | return quota_manager_.get(); | 
|  | } | 
|  |  | 
|  | TestBrowserContext* browser_context() { | 
|  | return browser_context_.get(); | 
|  | } | 
|  |  | 
|  | private: | 
|  | content::TestBrowserThreadBundle thread_bundle_; | 
|  | scoped_ptr<TestBrowserContext> browser_context_; | 
|  | scoped_refptr<MockQuotaManager> quota_manager_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(StoragePartitionImplTest); | 
|  | }; | 
|  |  | 
|  | class StoragePartitionShaderClearTest : public testing::Test { | 
|  | public: | 
|  | StoragePartitionShaderClearTest() | 
|  | : thread_bundle_(content::TestBrowserThreadBundle::IO_MAINLOOP), | 
|  | browser_context_(new TestBrowserContext()) { | 
|  | ShaderCacheFactory::GetInstance()->SetCacheInfo( | 
|  | kDefaultClientId, | 
|  | BrowserContext::GetDefaultStoragePartition( | 
|  | browser_context())->GetPath()); | 
|  | cache_ = ShaderCacheFactory::GetInstance()->Get(kDefaultClientId); | 
|  | } | 
|  |  | 
|  | virtual ~StoragePartitionShaderClearTest() { | 
|  | cache_ = NULL; | 
|  | ShaderCacheFactory::GetInstance()->RemoveCacheInfo(kDefaultClientId); | 
|  | } | 
|  |  | 
|  | void InitCache() { | 
|  | net::TestCompletionCallback available_cb; | 
|  | int rv = cache_->SetAvailableCallback(available_cb.callback()); | 
|  | ASSERT_EQ(net::OK, available_cb.GetResult(rv)); | 
|  | EXPECT_EQ(0, cache_->Size()); | 
|  |  | 
|  | cache_->Cache(kCacheKey, kCacheValue); | 
|  |  | 
|  | net::TestCompletionCallback complete_cb; | 
|  |  | 
|  | rv = cache_->SetCacheCompleteCallback(complete_cb.callback()); | 
|  | ASSERT_EQ(net::OK, complete_cb.GetResult(rv)); | 
|  | } | 
|  |  | 
|  | size_t Size() { return cache_->Size(); } | 
|  |  | 
|  | TestBrowserContext* browser_context() { | 
|  | return browser_context_.get(); | 
|  | } | 
|  |  | 
|  | private: | 
|  | content::TestBrowserThreadBundle thread_bundle_; | 
|  | scoped_ptr<TestBrowserContext> browser_context_; | 
|  |  | 
|  | scoped_refptr<ShaderDiskCache> cache_; | 
|  | }; | 
|  |  | 
|  | // Tests --------------------------------------------------------------------- | 
|  |  | 
|  | TEST_F(StoragePartitionShaderClearTest, ClearShaderCache) { | 
|  | InitCache(); | 
|  | EXPECT_EQ(1u, Size()); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind( | 
|  | &ClearData, | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context()), | 
|  | &run_loop)); | 
|  | run_loop.Run(); | 
|  | EXPECT_EQ(0u, Size()); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, QuotaClientMaskGeneration) { | 
|  | EXPECT_EQ(storage::QuotaClient::kFileSystem, | 
|  | StoragePartitionImpl::GenerateQuotaClientMask( | 
|  | StoragePartition::REMOVE_DATA_MASK_FILE_SYSTEMS)); | 
|  | EXPECT_EQ(storage::QuotaClient::kDatabase, | 
|  | StoragePartitionImpl::GenerateQuotaClientMask( | 
|  | StoragePartition::REMOVE_DATA_MASK_WEBSQL)); | 
|  | EXPECT_EQ(storage::QuotaClient::kAppcache, | 
|  | StoragePartitionImpl::GenerateQuotaClientMask( | 
|  | StoragePartition::REMOVE_DATA_MASK_APPCACHE)); | 
|  | EXPECT_EQ(storage::QuotaClient::kIndexedDatabase, | 
|  | StoragePartitionImpl::GenerateQuotaClientMask( | 
|  | StoragePartition::REMOVE_DATA_MASK_INDEXEDDB)); | 
|  | EXPECT_EQ(storage::QuotaClient::kFileSystem | | 
|  | storage::QuotaClient::kDatabase | | 
|  | storage::QuotaClient::kAppcache | | 
|  | storage::QuotaClient::kIndexedDatabase, | 
|  | StoragePartitionImpl::GenerateQuotaClientMask(kAllQuotaRemoveMask)); | 
|  | } | 
|  |  | 
|  | void PopulateTestQuotaManagedPersistentData(MockQuotaManager* manager) { | 
|  | manager->AddOrigin(kOrigin2, kPersistent, kClientFile, base::Time()); | 
|  | manager->AddOrigin(kOrigin3, kPersistent, kClientFile, | 
|  | base::Time::Now() - base::TimeDelta::FromDays(1)); | 
|  |  | 
|  | EXPECT_FALSE(manager->OriginHasData(kOrigin1, kPersistent, kClientFile)); | 
|  | EXPECT_TRUE(manager->OriginHasData(kOrigin2, kPersistent, kClientFile)); | 
|  | EXPECT_TRUE(manager->OriginHasData(kOrigin3, kPersistent, kClientFile)); | 
|  | } | 
|  |  | 
|  | void PopulateTestQuotaManagedTemporaryData(MockQuotaManager* manager) { | 
|  | manager->AddOrigin(kOrigin1, kTemporary, kClientFile, base::Time::Now()); | 
|  | manager->AddOrigin(kOrigin3, kTemporary, kClientFile, | 
|  | base::Time::Now() - base::TimeDelta::FromDays(1)); | 
|  |  | 
|  | EXPECT_TRUE(manager->OriginHasData(kOrigin1, kTemporary, kClientFile)); | 
|  | EXPECT_FALSE(manager->OriginHasData(kOrigin2, kTemporary, kClientFile)); | 
|  | EXPECT_TRUE(manager->OriginHasData(kOrigin3, kTemporary, kClientFile)); | 
|  | } | 
|  |  | 
|  | void PopulateTestQuotaManagedData(MockQuotaManager* manager) { | 
|  | // Set up kOrigin1 with a temporary quota, kOrigin2 with a persistent | 
|  | // quota, and kOrigin3 with both. kOrigin1 is modified now, kOrigin2 | 
|  | // is modified at the beginning of time, and kOrigin3 is modified one day | 
|  | // ago. | 
|  | PopulateTestQuotaManagedPersistentData(manager); | 
|  | PopulateTestQuotaManagedTemporaryData(manager); | 
|  | } | 
|  |  | 
|  | void PopulateTestQuotaManagedNonBrowsingData(MockQuotaManager* manager) { | 
|  | manager->AddOrigin(kOriginDevTools, kTemporary, kClientFile, base::Time()); | 
|  | manager->AddOrigin(kOriginDevTools, kPersistent, kClientFile, base::Time()); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveQuotaManagedDataForeverBoth) { | 
|  | PopulateTestQuotaManagedData(GetMockManager()); | 
|  |  | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideQuotaManagerForTesting( | 
|  | GetMockManager()); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind(&ClearQuotaData, partition, &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kPersistent, | 
|  | kClientFile)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveQuotaManagedDataForeverOnlyTemporary) { | 
|  | PopulateTestQuotaManagedTemporaryData(GetMockManager()); | 
|  |  | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideQuotaManagerForTesting( | 
|  | GetMockManager()); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind(&ClearQuotaData, partition, &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kPersistent, | 
|  | kClientFile)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveQuotaManagedDataForeverOnlyPersistent) { | 
|  | PopulateTestQuotaManagedPersistentData(GetMockManager()); | 
|  |  | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideQuotaManagerForTesting( | 
|  | GetMockManager()); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind(&ClearQuotaData, partition, &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kPersistent, | 
|  | kClientFile)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveQuotaManagedDataForeverNeither) { | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideQuotaManagerForTesting( | 
|  | GetMockManager()); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind(&ClearQuotaData, partition, &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kPersistent, | 
|  | kClientFile)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveQuotaManagedDataForeverSpecificOrigin) { | 
|  | PopulateTestQuotaManagedData(GetMockManager()); | 
|  |  | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideQuotaManagerForTesting( | 
|  | GetMockManager()); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind(&ClearQuotaDataForOrigin, | 
|  | partition, kOrigin1, base::Time(), | 
|  | &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin3, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin2, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin3, kPersistent, | 
|  | kClientFile)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveQuotaManagedDataForLastHour) { | 
|  | PopulateTestQuotaManagedData(GetMockManager()); | 
|  |  | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideQuotaManagerForTesting( | 
|  | GetMockManager()); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind(&ClearQuotaDataForOrigin, | 
|  | partition, GURL(), | 
|  | base::Time::Now() - base::TimeDelta::FromHours(1), | 
|  | &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin3, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin2, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin3, kPersistent, | 
|  | kClientFile)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveQuotaManagedDataForLastWeek) { | 
|  | PopulateTestQuotaManagedData(GetMockManager()); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideQuotaManagerForTesting( | 
|  | GetMockManager()); | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind(&ClearQuotaDataForNonPersistent, | 
|  | partition, | 
|  | base::Time::Now() - base::TimeDelta::FromDays(7), | 
|  | &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin2, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin3, kPersistent, | 
|  | kClientFile)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveQuotaManagedUnprotectedOrigins) { | 
|  | // Protect kOrigin1. | 
|  | scoped_refptr<MockSpecialStoragePolicy> mock_policy = | 
|  | new MockSpecialStoragePolicy; | 
|  | mock_policy->AddProtected(kOrigin1.GetOrigin()); | 
|  |  | 
|  | PopulateTestQuotaManagedData(GetMockManager()); | 
|  |  | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideQuotaManagerForTesting( | 
|  | GetMockManager()); | 
|  | partition->OverrideSpecialStoragePolicyForTesting(mock_policy); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind(&ClearQuotaDataWithOriginMatcher, | 
|  | partition, GURL(), | 
|  | base::Bind(&DoesOriginMatchForUnprotectedWeb), | 
|  | base::Time(), &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin1, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kPersistent, | 
|  | kClientFile)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveQuotaManagedProtectedSpecificOrigin) { | 
|  | // Protect kOrigin1. | 
|  | scoped_refptr<MockSpecialStoragePolicy> mock_policy = | 
|  | new MockSpecialStoragePolicy; | 
|  | mock_policy->AddProtected(kOrigin1.GetOrigin()); | 
|  |  | 
|  | PopulateTestQuotaManagedData(GetMockManager()); | 
|  |  | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideQuotaManagerForTesting( | 
|  | GetMockManager()); | 
|  | partition->OverrideSpecialStoragePolicyForTesting(mock_policy); | 
|  |  | 
|  | // Try to remove kOrigin1. Expect failure. | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind(&ClearQuotaDataWithOriginMatcher, | 
|  | partition, kOrigin1, | 
|  | base::Bind(&DoesOriginMatchForUnprotectedWeb), | 
|  | base::Time(), &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin1, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin3, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin2, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOrigin3, kPersistent, | 
|  | kClientFile)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveQuotaManagedProtectedOrigins) { | 
|  | // Protect kOrigin1. | 
|  | scoped_refptr<MockSpecialStoragePolicy> mock_policy = | 
|  | new MockSpecialStoragePolicy; | 
|  | mock_policy->AddProtected(kOrigin1.GetOrigin()); | 
|  |  | 
|  | PopulateTestQuotaManagedData(GetMockManager()); | 
|  |  | 
|  | // Try to remove kOrigin1. Expect success. | 
|  | base::RunLoop run_loop; | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideQuotaManagerForTesting( | 
|  | GetMockManager()); | 
|  | partition->OverrideSpecialStoragePolicyForTesting(mock_policy); | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, | 
|  | base::Bind(&ClearQuotaDataWithOriginMatcher, | 
|  | partition, GURL(), | 
|  | base::Bind(&DoesOriginMatchForBothProtectedAndUnprotectedWeb), | 
|  | base::Time(), &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin1, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin2, kPersistent, | 
|  | kClientFile)); | 
|  | EXPECT_FALSE(GetMockManager()->OriginHasData(kOrigin3, kPersistent, | 
|  | kClientFile)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveQuotaManagedIgnoreDevTools) { | 
|  | PopulateTestQuotaManagedNonBrowsingData(GetMockManager()); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideQuotaManagerForTesting( | 
|  | GetMockManager()); | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind(&ClearQuotaDataWithOriginMatcher, | 
|  | partition, GURL(), | 
|  | base::Bind(&DoesOriginMatchUnprotected), | 
|  | base::Time(), &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | // Check that devtools data isn't removed. | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOriginDevTools, kTemporary, | 
|  | kClientFile)); | 
|  | EXPECT_TRUE(GetMockManager()->OriginHasData(kOriginDevTools, kPersistent, | 
|  | kClientFile)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveCookieForever) { | 
|  | RemoveCookieTester tester(browser_context()); | 
|  |  | 
|  | tester.AddCookie(); | 
|  | ASSERT_TRUE(tester.ContainsCookie()); | 
|  |  | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->SetURLRequestContext(browser_context()->GetRequestContext()); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind(&ClearCookies, | 
|  | partition, base::Time(), base::Time::Max(), | 
|  | &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_FALSE(tester.ContainsCookie()); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveCookieLastHour) { | 
|  | RemoveCookieTester tester(browser_context()); | 
|  |  | 
|  | tester.AddCookie(); | 
|  | ASSERT_TRUE(tester.ContainsCookie()); | 
|  |  | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | base::Time an_hour_ago = base::Time::Now() - base::TimeDelta::FromHours(1); | 
|  | partition->SetURLRequestContext(browser_context()->GetRequestContext()); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, base::Bind(&ClearCookies, | 
|  | partition, an_hour_ago, base::Time::Max(), | 
|  | &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_FALSE(tester.ContainsCookie()); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveUnprotectedLocalStorageForever) { | 
|  | // Protect kOrigin1. | 
|  | scoped_refptr<MockSpecialStoragePolicy> mock_policy = | 
|  | new MockSpecialStoragePolicy; | 
|  | mock_policy->AddProtected(kOrigin1.GetOrigin()); | 
|  |  | 
|  | RemoveLocalStorageTester tester(browser_context()); | 
|  |  | 
|  | tester.AddDOMStorageTestData(); | 
|  | EXPECT_TRUE(tester.DOMStorageExistsForOrigin(kOrigin1)); | 
|  | EXPECT_TRUE(tester.DOMStorageExistsForOrigin(kOrigin2)); | 
|  | EXPECT_TRUE(tester.DOMStorageExistsForOrigin(kOrigin3)); | 
|  |  | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideSpecialStoragePolicyForTesting(mock_policy); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, | 
|  | base::Bind(&ClearStuff, | 
|  | StoragePartitionImpl::REMOVE_DATA_MASK_LOCAL_STORAGE, | 
|  | partition, base::Time(), base::Time::Max(), | 
|  | base::Bind(&DoesOriginMatchForUnprotectedWeb), | 
|  | &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_TRUE(tester.DOMStorageExistsForOrigin(kOrigin1)); | 
|  | EXPECT_FALSE(tester.DOMStorageExistsForOrigin(kOrigin2)); | 
|  | EXPECT_FALSE(tester.DOMStorageExistsForOrigin(kOrigin3)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveProtectedLocalStorageForever) { | 
|  | // Protect kOrigin1. | 
|  | scoped_refptr<MockSpecialStoragePolicy> mock_policy = | 
|  | new MockSpecialStoragePolicy; | 
|  | mock_policy->AddProtected(kOrigin1.GetOrigin()); | 
|  |  | 
|  | RemoveLocalStorageTester tester(browser_context()); | 
|  |  | 
|  | tester.AddDOMStorageTestData(); | 
|  | EXPECT_TRUE(tester.DOMStorageExistsForOrigin(kOrigin1)); | 
|  | EXPECT_TRUE(tester.DOMStorageExistsForOrigin(kOrigin2)); | 
|  | EXPECT_TRUE(tester.DOMStorageExistsForOrigin(kOrigin3)); | 
|  |  | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | partition->OverrideSpecialStoragePolicyForTesting(mock_policy); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, | 
|  | base::Bind(&ClearStuff, | 
|  | StoragePartitionImpl::REMOVE_DATA_MASK_LOCAL_STORAGE, | 
|  | partition, base::Time(), base::Time::Max(), | 
|  | base::Bind(&DoesOriginMatchForBothProtectedAndUnprotectedWeb), | 
|  | &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | // Even if kOrigin1 is protected, it will be deleted since we specify | 
|  | // ClearData to delete protected data. | 
|  | EXPECT_FALSE(tester.DOMStorageExistsForOrigin(kOrigin1)); | 
|  | EXPECT_FALSE(tester.DOMStorageExistsForOrigin(kOrigin2)); | 
|  | EXPECT_FALSE(tester.DOMStorageExistsForOrigin(kOrigin3)); | 
|  | } | 
|  |  | 
|  | TEST_F(StoragePartitionImplTest, RemoveLocalStorageForLastWeek) { | 
|  | RemoveLocalStorageTester tester(browser_context()); | 
|  |  | 
|  | tester.AddDOMStorageTestData(); | 
|  | EXPECT_TRUE(tester.DOMStorageExistsForOrigin(kOrigin1)); | 
|  | EXPECT_TRUE(tester.DOMStorageExistsForOrigin(kOrigin2)); | 
|  | EXPECT_TRUE(tester.DOMStorageExistsForOrigin(kOrigin3)); | 
|  |  | 
|  | StoragePartitionImpl* partition = static_cast<StoragePartitionImpl*>( | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context())); | 
|  | base::Time a_week_ago = base::Time::Now() - base::TimeDelta::FromDays(7); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | base::MessageLoop::current()->PostTask( | 
|  | FROM_HERE, | 
|  | base::Bind(&ClearStuff, | 
|  | StoragePartitionImpl::REMOVE_DATA_MASK_LOCAL_STORAGE, | 
|  | partition, a_week_ago, base::Time::Max(), | 
|  | base::Bind(&DoesOriginMatchForBothProtectedAndUnprotectedWeb), | 
|  | &run_loop)); | 
|  | run_loop.Run(); | 
|  |  | 
|  | // kOrigin1 and kOrigin2 do not have age more than a week. | 
|  | EXPECT_FALSE(tester.DOMStorageExistsForOrigin(kOrigin1)); | 
|  | EXPECT_FALSE(tester.DOMStorageExistsForOrigin(kOrigin2)); | 
|  | EXPECT_TRUE(tester.DOMStorageExistsForOrigin(kOrigin3)); | 
|  | } | 
|  |  | 
|  | }  // namespace content |