blob: e1497f57321e71a53f750927bb1d2b6ea239ebe5 [file] [log] [blame]
// 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());
}