| // Copyright 2012 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "net/disk_cache/disk_cache_test_base.h" |
| |
| #include <memory> |
| #include <utility> |
| |
| #include "base/files/file_util.h" |
| #include "base/functional/bind.h" |
| #include "base/notreached.h" |
| #include "base/path_service.h" |
| #include "base/run_loop.h" |
| #include "base/task/single_thread_task_runner.h" |
| #include "base/threading/platform_thread.h" |
| #include "net/base/io_buffer.h" |
| #include "net/base/net_errors.h" |
| #include "net/base/request_priority.h" |
| #include "net/base/test_completion_callback.h" |
| #include "net/disk_cache/backend_cleanup_tracker.h" |
| #include "net/disk_cache/blockfile/backend_impl.h" |
| #include "net/disk_cache/cache_util.h" |
| #include "net/disk_cache/disk_cache.h" |
| #include "net/disk_cache/disk_cache_test_util.h" |
| #include "net/disk_cache/memory/mem_backend_impl.h" |
| #include "net/disk_cache/simple/simple_backend_impl.h" |
| #include "net/disk_cache/simple/simple_file_tracker.h" |
| #include "net/disk_cache/simple/simple_index.h" |
| #include "net/test/gtest_util.h" |
| #include "testing/gmock/include/gmock/gmock.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| #if BUILDFLAG(ENABLE_DISK_CACHE_SQL_BACKEND) |
| #include "net/disk_cache/sql/sql_backend_impl.h" |
| #endif // ENABLE_DISK_CACHE_SQL_BACKEND |
| |
| using net::test::IsOk; |
| |
| DiskCacheTest::DiskCacheTest( |
| base::test::TaskEnvironment::TimeSource time_source) |
| : WithTaskEnvironment(time_source) { |
| CHECK(temp_dir_.CreateUniqueTempDir()); |
| // Put the cache into a subdir of |temp_dir_|, to permit tests to safely |
| // remove the cache directory without risking collisions with other tests. |
| cache_path_ = temp_dir_.GetPath().AppendASCII("cache"); |
| CHECK(base::CreateDirectory(cache_path_)); |
| } |
| |
| DiskCacheTest::~DiskCacheTest() = default; |
| |
| bool DiskCacheTest::CopyTestCache(const std::string& name) { |
| base::FilePath path; |
| base::PathService::Get(base::DIR_SRC_TEST_DATA_ROOT, &path); |
| path = path.AppendASCII("net"); |
| path = path.AppendASCII("data"); |
| path = path.AppendASCII("cache_tests"); |
| path = path.AppendASCII(name); |
| |
| if (!CleanupCacheDir()) |
| return false; |
| return base::CopyDirectory(path, cache_path_, false); |
| } |
| |
| bool DiskCacheTest::CleanupCacheDir() { |
| return DeleteCache(cache_path_); |
| } |
| |
| void DiskCacheTest::TearDown() { |
| RunUntilIdle(); |
| } |
| |
| // static |
| std::string DiskCacheTestWithCache::BackendToTestName( |
| BackendToTest backend_to_test) { |
| switch (backend_to_test) { |
| case BackendToTest::kBlockfile: |
| return "Blockfile"; |
| case BackendToTest::kSimple: |
| return "Simple"; |
| case BackendToTest::kMemory: |
| return "Memory"; |
| #if BUILDFLAG(ENABLE_DISK_CACHE_SQL_BACKEND) |
| case BackendToTest::kSql: |
| return "Sql"; |
| #endif // ENABLE_DISK_CACHE_SQL_BACKEND |
| } |
| NOTREACHED(); |
| } |
| |
| DiskCacheTestWithCache::TestIterator::TestIterator( |
| std::unique_ptr<disk_cache::Backend::Iterator> iterator) |
| : iterator_(std::move(iterator)) {} |
| |
| DiskCacheTestWithCache::TestIterator::~TestIterator() = default; |
| |
| int DiskCacheTestWithCache::TestIterator::OpenNextEntry( |
| disk_cache::Entry** next_entry) { |
| TestEntryResultCompletionCallback cb; |
| disk_cache::EntryResult result = |
| cb.GetResult(iterator_->OpenNextEntry(cb.callback())); |
| int rv = result.net_error(); |
| *next_entry = result.ReleaseEntry(); |
| return rv; |
| } |
| |
| DiskCacheTestWithCache::DiskCacheTestWithCache( |
| base::test::TaskEnvironment::TimeSource time_source) |
| : DiskCacheTest(time_source) {} |
| |
| DiskCacheTestWithCache::~DiskCacheTestWithCache() = default; |
| |
| void DiskCacheTestWithCache::InitCache() { |
| if (backend_to_test_ == BackendToTest::kMemory) { |
| InitMemoryCache(); |
| } else { |
| InitDiskCache(); |
| } |
| |
| ASSERT_TRUE(nullptr != cache_); |
| if (first_cleanup_) |
| ASSERT_EQ(0, GetEntryCount()); |
| } |
| |
| // We are expected to leak memory when simulating crashes. |
| void DiskCacheTestWithCache::SimulateCrash() { |
| ASSERT_EQ(backend_to_test_, BackendToTest::kBlockfile); |
| net::TestCompletionCallback cb; |
| int rv = cache_impl_->FlushQueueForTest(cb.callback()); |
| ASSERT_THAT(cb.GetResult(rv), IsOk()); |
| cache_impl_->ClearRefCountForTest(); |
| |
| ResetCaches(); |
| EXPECT_TRUE(CheckCacheIntegrity(cache_path_, new_eviction_, size_, mask_)); |
| |
| CreateBackend(disk_cache::kNoRandom); |
| } |
| |
| void DiskCacheTestWithCache::SetTestMode() { |
| ASSERT_EQ(backend_to_test_, BackendToTest::kBlockfile); |
| cache_impl_->SetUnitTestMode(); |
| } |
| |
| void DiskCacheTestWithCache::SetMaxSize(int64_t size) { |
| size_ = size; |
| // Cache size should not generally be changed dynamically; it takes |
| // backend-specific knowledge to make it even semi-reasonable to do. |
| DCHECK(!cache_); |
| } |
| |
| int32_t DiskCacheTestWithCache::GetEntryCount() { |
| net::TestInt32CompletionCallback cb; |
| return cb.GetResult(cache_->GetEntryCount(cb.callback())); |
| } |
| |
| disk_cache::EntryResult DiskCacheTestWithCache::OpenOrCreateEntry( |
| const std::string& key) { |
| return OpenOrCreateEntryWithPriority(key, net::HIGHEST); |
| } |
| |
| disk_cache::EntryResult DiskCacheTestWithCache::OpenOrCreateEntryWithPriority( |
| const std::string& key, |
| net::RequestPriority request_priority) { |
| TestEntryResultCompletionCallback cb; |
| disk_cache::EntryResult result = |
| cache_->OpenOrCreateEntry(key, request_priority, cb.callback()); |
| return cb.GetResult(std::move(result)); |
| } |
| |
| int DiskCacheTestWithCache::OpenEntry(const std::string& key, |
| disk_cache::Entry** entry) { |
| return OpenEntryWithPriority(key, net::HIGHEST, entry); |
| } |
| |
| int DiskCacheTestWithCache::OpenEntryWithPriority( |
| const std::string& key, |
| net::RequestPriority request_priority, |
| disk_cache::Entry** entry) { |
| TestEntryResultCompletionCallback cb; |
| disk_cache::EntryResult result = |
| cb.GetResult(cache_->OpenEntry(key, request_priority, cb.callback())); |
| int rv = result.net_error(); |
| *entry = result.ReleaseEntry(); |
| return rv; |
| } |
| |
| int DiskCacheTestWithCache::CreateEntry(const std::string& key, |
| disk_cache::Entry** entry) { |
| return CreateEntryWithPriority(key, net::HIGHEST, entry); |
| } |
| |
| int DiskCacheTestWithCache::CreateEntryWithPriority( |
| const std::string& key, |
| net::RequestPriority request_priority, |
| disk_cache::Entry** entry) { |
| TestEntryResultCompletionCallback cb; |
| disk_cache::EntryResult result = |
| cb.GetResult(cache_->CreateEntry(key, request_priority, cb.callback())); |
| int rv = result.net_error(); |
| *entry = result.ReleaseEntry(); |
| return rv; |
| } |
| |
| int DiskCacheTestWithCache::DoomEntry(const std::string& key) { |
| net::TestCompletionCallback cb; |
| int rv = cache_->DoomEntry(key, net::HIGHEST, cb.callback()); |
| return cb.GetResult(rv); |
| } |
| |
| int DiskCacheTestWithCache::DoomAllEntries() { |
| net::TestCompletionCallback cb; |
| int rv = cache_->DoomAllEntries(cb.callback()); |
| return cb.GetResult(rv); |
| } |
| |
| int DiskCacheTestWithCache::DoomEntriesBetween(const base::Time initial_time, |
| const base::Time end_time) { |
| net::TestCompletionCallback cb; |
| int rv = cache_->DoomEntriesBetween(initial_time, end_time, cb.callback()); |
| return cb.GetResult(rv); |
| } |
| |
| int DiskCacheTestWithCache::DoomEntriesSince(const base::Time initial_time) { |
| net::TestCompletionCallback cb; |
| int rv = cache_->DoomEntriesSince(initial_time, cb.callback()); |
| return cb.GetResult(rv); |
| } |
| |
| int64_t DiskCacheTestWithCache::CalculateSizeOfAllEntries() { |
| net::TestInt64CompletionCallback cb; |
| int64_t rv = cache_->CalculateSizeOfAllEntries(cb.callback()); |
| return cb.GetResult(rv); |
| } |
| |
| int64_t DiskCacheTestWithCache::CalculateSizeOfEntriesBetween( |
| const base::Time initial_time, |
| const base::Time end_time) { |
| net::TestInt64CompletionCallback cb; |
| int64_t rv = cache_->CalculateSizeOfEntriesBetween(initial_time, end_time, |
| cb.callback()); |
| return cb.GetResult(rv); |
| } |
| |
| std::unique_ptr<DiskCacheTestWithCache::TestIterator> |
| DiskCacheTestWithCache::CreateIterator() { |
| return std::make_unique<TestIterator>(cache_->CreateIterator()); |
| } |
| |
| void DiskCacheTestWithCache::FlushQueueForTest() { |
| if (backend_to_test_ == BackendToTest::kMemory) { |
| // No threading to flush. |
| return; |
| } |
| |
| if (simple_cache_impl_) { |
| disk_cache::FlushCacheThreadForTesting(); |
| return; |
| } |
| |
| #if BUILDFLAG(ENABLE_DISK_CACHE_SQL_BACKEND) |
| if (sql_cache_impl_) { |
| net::TestCompletionCallback cb; |
| int rv = sql_cache_impl_->FlushQueueForTest(cb.callback()); |
| EXPECT_THAT(cb.GetResult(rv), IsOk()); |
| return; |
| } |
| #endif // ENABLE_DISK_CACHE_SQL_BACKEND |
| |
| DCHECK(cache_impl_); |
| net::TestCompletionCallback cb; |
| int rv = cache_impl_->FlushQueueForTest(cb.callback()); |
| EXPECT_THAT(cb.GetResult(rv), IsOk()); |
| } |
| |
| void DiskCacheTestWithCache::RunTaskForTest(base::OnceClosure closure) { |
| if (backend_to_test_ == BackendToTest::kMemory) { |
| // For memory backend, cache thread is always just current thread,s o |
| // we can run the task directly. |
| std::move(closure).Run(); |
| return; |
| } |
| // Blockfile backend provides a way of running tasks on its work thread; |
| // the notion doesn't make sense for simple. |
| CHECK_EQ(backend_to_test_, BackendToTest::kBlockfile); |
| |
| net::TestCompletionCallback cb; |
| int rv = cache_impl_->RunTaskForTest(std::move(closure), cb.callback()); |
| EXPECT_THAT(cb.GetResult(rv), IsOk()); |
| } |
| |
| int DiskCacheTestWithCache::ReadData(disk_cache::Entry* entry, |
| int index, |
| int offset, |
| net::IOBuffer* buf, |
| int len) { |
| net::TestCompletionCallback cb; |
| int rv = entry->ReadData(index, offset, buf, len, cb.callback()); |
| return cb.GetResult(rv); |
| } |
| |
| int DiskCacheTestWithCache::WriteData(disk_cache::Entry* entry, |
| int index, |
| int offset, |
| net::IOBuffer* buf, |
| int len, |
| bool truncate) { |
| net::TestCompletionCallback cb; |
| int rv = entry->WriteData(index, offset, buf, len, cb.callback(), truncate); |
| return cb.GetResult(rv); |
| } |
| |
| int DiskCacheTestWithCache::ReadSparseData(disk_cache::Entry* entry, |
| int64_t offset, |
| net::IOBuffer* buf, |
| int len) { |
| net::TestCompletionCallback cb; |
| int rv = entry->ReadSparseData(offset, buf, len, cb.callback()); |
| return cb.GetResult(rv); |
| } |
| |
| int DiskCacheTestWithCache::WriteSparseData(disk_cache::Entry* entry, |
| int64_t offset, |
| net::IOBuffer* buf, |
| int len) { |
| net::TestCompletionCallback cb; |
| int rv = entry->WriteSparseData(offset, buf, len, cb.callback()); |
| return cb.GetResult(rv); |
| } |
| |
| int DiskCacheTestWithCache::GetAvailableRange(disk_cache::Entry* entry, |
| int64_t offset, |
| int len, |
| int64_t* start) { |
| TestRangeResultCompletionCallback cb; |
| disk_cache::RangeResult result = |
| cb.GetResult(entry->GetAvailableRange(offset, len, cb.callback())); |
| |
| if (result.net_error == net::OK) { |
| *start = result.start; |
| return result.available_len; |
| } |
| return result.net_error; |
| } |
| |
| void DiskCacheTestWithCache::TrimForTest(bool empty) { |
| CHECK_EQ(backend_to_test_, BackendToTest::kBlockfile); |
| |
| RunTaskForTest(base::BindOnce(&disk_cache::BackendImpl::TrimForTest, |
| base::Unretained(cache_impl_), empty)); |
| } |
| |
| void DiskCacheTestWithCache::TrimDeletedListForTest(bool empty) { |
| CHECK_EQ(backend_to_test_, BackendToTest::kBlockfile); |
| |
| RunTaskForTest( |
| base::BindOnce(&disk_cache::BackendImpl::TrimDeletedListForTest, |
| base::Unretained(cache_impl_), empty)); |
| } |
| |
| void DiskCacheTestWithCache::AddDelay() { |
| // Advance time by 1 second. This ensures that time-sensitive operations, |
| // particularly those in Simple Cache which has second-level timestamp |
| // granularity, will see a change in time. |
| FastForwardBy(base::Seconds(1)); |
| } |
| |
| void DiskCacheTestWithCache::OnExternalCacheHit(const std::string& key) { |
| cache_->OnExternalCacheHit(key); |
| } |
| |
| std::unique_ptr<disk_cache::Backend> DiskCacheTestWithCache::TakeCache() { |
| mem_cache_ = nullptr; |
| simple_cache_impl_ = nullptr; |
| #if BUILDFLAG(ENABLE_DISK_CACHE_SQL_BACKEND) |
| sql_cache_impl_ = nullptr; |
| #endif // ENABLE_DISK_CACHE_SQL_BACKEND |
| cache_impl_ = nullptr; |
| return std::move(cache_); |
| } |
| |
| void DiskCacheTestWithCache::TearDown() { |
| RunUntilIdle(); |
| ResetCaches(); |
| if (backend_to_test_ == BackendToTest::kBlockfile && integrity_) { |
| EXPECT_TRUE(CheckCacheIntegrity(cache_path_, new_eviction_, size_, mask_)); |
| } |
| RunUntilIdle(); |
| if (backend_to_test_ == BackendToTest::kSimple && simple_file_tracker_) { |
| EXPECT_TRUE(simple_file_tracker_->IsEmptyForTesting()); |
| } |
| DiskCacheTest::TearDown(); |
| } |
| |
| void DiskCacheTestWithCache::ResetCaches() { |
| #if BUILDFLAG(ENABLE_DISK_CACHE_SQL_BACKEND) |
| scoped_refptr<base::SequencedTaskRunner> background_task_runner; |
| if (sql_cache_impl_) { |
| background_task_runner = sql_cache_impl_->GetBackgroundTaskRunnerForTest(); |
| } |
| #endif // ENABLE_DISK_CACHE_SQL_BACKEND |
| std::unique_ptr<disk_cache::Backend> cache = TakeCache(); |
| cache.reset(); |
| #if BUILDFLAG(ENABLE_DISK_CACHE_SQL_BACKEND) |
| if (background_task_runner) { |
| base::RunLoop run_loop; |
| background_task_runner->PostTask(FROM_HERE, run_loop.QuitClosure()); |
| run_loop.Run(); |
| } |
| #endif // ENABLE_DISK_CACHE_SQL_BACKEND |
| } |
| |
| void DiskCacheTestWithCache::InitMemoryCache() { |
| auto cache = |
| disk_cache::MemBackendImpl::CreateBackend(size_, /*net_log=*/nullptr); |
| mem_cache_ = cache.get(); |
| cache_ = std::move(cache); |
| ASSERT_TRUE(cache_); |
| } |
| |
| void DiskCacheTestWithCache::InitDiskCache() { |
| if (first_cleanup_) |
| ASSERT_TRUE(CleanupCacheDir()); |
| |
| CreateBackend(disk_cache::kNoRandom); |
| } |
| |
| void DiskCacheTestWithCache::CreateBackend(uint32_t flags) { |
| scoped_refptr<base::SingleThreadTaskRunner> runner; |
| if (use_current_thread_) |
| runner = base::SingleThreadTaskRunner::GetCurrentDefault(); |
| else |
| runner = nullptr; // let the backend sort it out. |
| |
| if (backend_to_test_ == BackendToTest::kSimple) { |
| DCHECK(!use_current_thread_) |
| << "Using current thread unsupported by SimpleCache"; |
| net::TestCompletionCallback cb; |
| // We limit ourselves to 64 fds since OS X by default gives us 256. |
| // (Chrome raises the number on startup, but the test fixture doesn't). |
| if (!simple_file_tracker_) |
| simple_file_tracker_ = |
| std::make_unique<disk_cache::SimpleFileTracker>(64); |
| std::unique_ptr<disk_cache::SimpleBackendImpl> simple_backend = |
| std::make_unique<disk_cache::SimpleBackendImpl>( |
| /*file_operations=*/nullptr, cache_path_, |
| /* cleanup_tracker = */ nullptr, simple_file_tracker_.get(), size_, |
| type_, /*net_log = */ nullptr); |
| simple_backend->Init(cb.callback()); |
| ASSERT_THAT(cb.WaitForResult(), IsOk()); |
| simple_cache_impl_ = simple_backend.get(); |
| cache_ = std::move(simple_backend); |
| if (simple_cache_wait_for_index_) { |
| net::TestCompletionCallback wait_for_index_cb; |
| simple_cache_impl_->index()->ExecuteWhenReady( |
| wait_for_index_cb.callback()); |
| int rv = wait_for_index_cb.WaitForResult(); |
| ASSERT_THAT(rv, IsOk()); |
| } |
| return; |
| } |
| #if BUILDFLAG(ENABLE_DISK_CACHE_SQL_BACKEND) |
| if (backend_to_test_ == BackendToTest::kSql) { |
| net::TestCompletionCallback cb; |
| auto sql_backend = |
| std::make_unique<disk_cache::SqlBackendImpl>(cache_path_, size_, type_); |
| sql_backend->Init(cb.callback()); |
| ASSERT_THAT(cb.WaitForResult(), IsOk()); |
| sql_cache_impl_ = sql_backend.get(); |
| cache_ = std::move(sql_backend); |
| return; |
| } |
| #endif // ENABLE_DISK_CACHE_SQL_BACKEND |
| CHECK_EQ(backend_to_test_, BackendToTest::kBlockfile); |
| |
| std::unique_ptr<disk_cache::BackendImpl> cache; |
| if (mask_) { |
| cache = std::make_unique<disk_cache::BackendImpl>( |
| cache_path_, mask_, |
| /* cleanup_tracker = */ nullptr, runner, type_, |
| /* net_log = */ nullptr); |
| } else { |
| cache = std::make_unique<disk_cache::BackendImpl>( |
| cache_path_, /* cleanup_tracker = */ nullptr, runner, type_, |
| /* net_log = */ nullptr); |
| } |
| cache_impl_ = cache.get(); |
| cache_ = std::move(cache); |
| ASSERT_TRUE(cache_); |
| if (size_) |
| EXPECT_TRUE(cache_impl_->SetMaxSize(size_)); |
| if (new_eviction_) |
| cache_impl_->SetNewEviction(); |
| cache_impl_->SetFlags(flags); |
| net::TestCompletionCallback cb; |
| cache_impl_->Init(cb.callback()); |
| ASSERT_THAT(cb.WaitForResult(), IsOk()); |
| } |