|  | // 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()); | 
|  | } |