// Copyright 2011 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include #include #include "base/files/file_enumerator.h" #include "base/files/file_util.h" #include "base/files/safe_base_name.h" #include "base/files/scoped_temp_dir.h" #include "base/run_loop.h" #include "base/strings/string_number_conversions.h" #include "base/test/bind.h" #include "base/test/scoped_feature_list.h" #include "base/test/task_environment.h" #include "base/threading/platform_thread.h" #include "build/chromeos_buildflags.h" #include "net/disk_cache/cache_util.h" #include "testing/gtest/include/gtest/gtest.h" #include "testing/platform_test.h" namespace disk_cache { class CacheUtilTest : public PlatformTest { public: void SetUp() override { PlatformTest::SetUp(); ASSERT_TRUE(tmp_dir_.CreateUniqueTempDir()); cache_dir_ = tmp_dir_.GetPath().Append(FILE_PATH_LITERAL("Cache")); file1_ = base::FilePath(cache_dir_.Append(FILE_PATH_LITERAL("file01"))); file2_ = base::FilePath(cache_dir_.Append(FILE_PATH_LITERAL(".file02"))); dir1_ = base::FilePath(cache_dir_.Append(FILE_PATH_LITERAL("dir01"))); file3_ = base::FilePath(dir1_.Append(FILE_PATH_LITERAL("file03"))); ASSERT_TRUE(base::CreateDirectory(cache_dir_)); FILE *fp = base::OpenFile(file1_, "w"); ASSERT_TRUE(fp != nullptr); base::CloseFile(fp); fp = base::OpenFile(file2_, "w"); ASSERT_TRUE(fp != nullptr); base::CloseFile(fp); ASSERT_TRUE(base::CreateDirectory(dir1_)); fp = base::OpenFile(file3_, "w"); ASSERT_TRUE(fp != nullptr); base::CloseFile(fp); dest_dir_ = tmp_dir_.GetPath().Append(FILE_PATH_LITERAL("old_Cache_001")); dest_file1_ = base::FilePath(dest_dir_.Append(FILE_PATH_LITERAL("file01"))); dest_file2_ = base::FilePath(dest_dir_.Append(FILE_PATH_LITERAL(".file02"))); dest_dir1_ = base::FilePath(dest_dir_.Append(FILE_PATH_LITERAL("dir01"))); } protected: base::ScopedTempDir tmp_dir_; base::FilePath cache_dir_; base::FilePath file1_; base::FilePath file2_; base::FilePath dir1_; base::FilePath file3_; base::FilePath dest_dir_; base::FilePath dest_file1_; base::FilePath dest_file2_; base::FilePath dest_dir1_; base::test::TaskEnvironment task_environment_; }; TEST_F(CacheUtilTest, MoveCache) { EXPECT_TRUE(disk_cache::MoveCache(cache_dir_, dest_dir_)); EXPECT_TRUE(base::PathExists(dest_dir_)); EXPECT_TRUE(base::PathExists(dest_file1_)); EXPECT_TRUE(base::PathExists(dest_file2_)); EXPECT_TRUE(base::PathExists(dest_dir1_)); #if BUILDFLAG(IS_CHROMEOS_ASH) EXPECT_TRUE(base::PathExists(cache_dir_)); // old cache dir stays #else EXPECT_FALSE(base::PathExists(cache_dir_)); // old cache is gone #endif EXPECT_FALSE(base::PathExists(file1_)); EXPECT_FALSE(base::PathExists(file2_)); EXPECT_FALSE(base::PathExists(dir1_)); } TEST_F(CacheUtilTest, DeleteCache) { disk_cache::DeleteCache(cache_dir_, false); EXPECT_TRUE(base::PathExists(cache_dir_)); // cache dir stays EXPECT_FALSE(base::PathExists(dir1_)); EXPECT_FALSE(base::PathExists(file1_)); EXPECT_FALSE(base::PathExists(file2_)); EXPECT_FALSE(base::PathExists(file3_)); } TEST_F(CacheUtilTest, DeleteCacheAndDir) { disk_cache::DeleteCache(cache_dir_, true); EXPECT_FALSE(base::PathExists(cache_dir_)); // cache dir is gone EXPECT_FALSE(base::PathExists(dir1_)); EXPECT_FALSE(base::PathExists(file1_)); EXPECT_FALSE(base::PathExists(file2_)); EXPECT_FALSE(base::PathExists(file3_)); } TEST_F(CacheUtilTest, CleanupDirectory) { base::RunLoop run_loop; disk_cache::CleanupDirectory(cache_dir_, base::BindLambdaForTesting([&](bool result) { EXPECT_TRUE(result); run_loop.Quit(); })); run_loop.Run(); while (true) { base::FileEnumerator enumerator(tmp_dir_.GetPath(), /*recursive=*/false, /*file_type=*/base::FileEnumerator::FILES | base::FileEnumerator::DIRECTORIES); bool found = false; while (true) { base::FilePath path = enumerator.Next(); if (path.empty()) { break; } // We're not sure if we see an entry in the directory because it depends // on timing, but if we do, it must be "old_Cache_000". // Caveat: On ChromeOS, we leave the top-level directory ("Cache") so // it must be "Cache" or "old_Cache_000". const base::FilePath dirname = path.DirName(); absl::optional basename = base::SafeBaseName::Create(path); ASSERT_EQ(dirname, tmp_dir_.GetPath()); ASSERT_TRUE(basename.has_value()); #if BUILDFLAG(IS_CHROMEOS_ASH) if (basename->path().value() == FILE_PATH_LITERAL("Cache")) { // See the comment above. ASSERT_TRUE(base::IsDirectoryEmpty(dirname.Append(*basename))); continue; } #endif ASSERT_EQ(basename->path().value(), FILE_PATH_LITERAL("old_Cache_000")); found = true; } if (!found) { break; } base::PlatformThread::Sleep(base::Milliseconds(10)); } } #if BUILDFLAG(IS_POSIX) TEST_F(CacheUtilTest, CleanupDirectoryFailsWhenParentDirectoryIsInaccessible) { base::RunLoop run_loop; ASSERT_TRUE(base::SetPosixFilePermissions(tmp_dir_.GetPath(), /*mode=*/0)); disk_cache::CleanupDirectory(cache_dir_, base::BindLambdaForTesting([&](bool result) { EXPECT_FALSE(result); run_loop.Quit(); })); run_loop.Run(); } TEST_F(CacheUtilTest, CleanupDirectorySucceedsWhenTargetDirectoryIsInaccessible) { base::RunLoop run_loop; ASSERT_TRUE(base::SetPosixFilePermissions(cache_dir_, /*mode=*/0)); disk_cache::CleanupDirectory(cache_dir_, base::BindLambdaForTesting([&](bool result) { EXPECT_TRUE(result); run_loop.Quit(); })); run_loop.Run(); } #endif TEST_F(CacheUtilTest, PreferredCacheSize) { const struct TestCase { int64_t available; int expected_without_trial; int expected_with_200_trial; int expected_with_250_trial; int expected_with_300_trial; } kTestCases[] = { // Weird negative value for available --- return the "default" {-1000LL, 80 * 1024 * 1024, 160 * 1024 * 1024, 200 * 1024 * 1024, 240 * 1024 * 1024}, {-1LL, 80 * 1024 * 1024, 160 * 1024 * 1024, 200 * 1024 * 1024, 240 * 1024 * 1024}, // 0 produces 0. {0LL, 0, 0, 0, 0}, // Cache is 80% of available space, when default cache size is larger than // 80% of available space.. {50 * 1024 * 1024LL, 40 * 1024 * 1024, 40 * 1024 * 1024, 40 * 1024 * 1024, 40 * 1024 * 1024}, // Cache is default size, when default size is 10% to 80% of available // space. {100 * 1024 * 1024LL, 80 * 1024 * 1024, 80 * 1024 * 1024, 80 * 1024 * 1024, 80 * 1024 * 1024}, {200 * 1024 * 1024LL, 80 * 1024 * 1024, 80 * 1024 * 1024, 80 * 1024 * 1024, 80 * 1024 * 1024}, // Cache is 10% of available space if 2.5 * default size is more than 10% // of available space. {1000 * 1024 * 1024LL, 100 * 1024 * 1024, 200 * 1024 * 1024, 200 * 1024 * 1024, 200 * 1024 * 1024}, {2000 * 1024 * 1024LL, 200 * 1024 * 1024, 400 * 1024 * 1024, 400 * 1024 * 1024, 400 * 1024 * 1024}, // Cache is 2.5 * kDefaultCacheSize if 2.5 * kDefaultCacheSize uses from // 1% to 10% of available space. {10000 * 1024 * 1024LL, 200 * 1024 * 1024, 400 * 1024 * 1024, 500 * 1024 * 1024, 600 * 1024 * 1024}, // Otherwise, cache is 1% of available space. {20000 * 1024 * 1024LL, 200 * 1024 * 1024, 400 * 1024 * 1024, 500 * 1024 * 1024, 600 * 1024 * 1024}, // Until it runs into the cache size cap. {32000 * 1024 * 1024LL, 320 * 1024 * 1024, 640 * 1024 * 1024, 800 * 1024 * 1024, 960 * 1024 * 1024}, {50000 * 1024 * 1024LL, 320 * 1024 * 1024, 640 * 1024 * 1024, 800 * 1024 * 1024, 960 * 1024 * 1024}, }; for (const auto& test_case : kTestCases) { EXPECT_EQ(test_case.expected_without_trial, PreferredCacheSize(test_case.available)) << test_case.available; // Preferred size for WebUI code cache matches expected_without_trial but // should never be more than 5 MB. int expected_webui_code_cache_size = std::min(5 * 1024 * 1024, test_case.expected_without_trial); EXPECT_EQ(expected_webui_code_cache_size, PreferredCacheSize(test_case.available, net::GENERATED_WEBUI_BYTE_CODE_CACHE)) << test_case.available; } // Check that the cache size cap is 50% higher for native code caches. EXPECT_EQ(((320 * 1024 * 1024) / 2) * 3, PreferredCacheSize(50000 * 1024 * 1024LL, net::GENERATED_NATIVE_CODE_CACHE)); for (int cache_size_exeriment : {100, 200, 250, 300}) { base::test::ScopedFeatureList scoped_feature_list; std::map field_trial_params; field_trial_params["percent_relative_size"] = base::NumberToString(cache_size_exeriment); scoped_feature_list.InitAndEnableFeatureWithParameters( disk_cache::kChangeDiskCacheSizeExperiment, field_trial_params); for (const auto& test_case : kTestCases) { int expected = 0; switch (cache_size_exeriment) { case 100: expected = test_case.expected_without_trial; break; case 200: expected = test_case.expected_with_200_trial; break; case 250: expected = test_case.expected_with_250_trial; break; case 300: expected = test_case.expected_with_300_trial; break; } EXPECT_EQ(expected, PreferredCacheSize(test_case.available)); // For caches other than disk cache, the size is not scaled. EXPECT_EQ(test_case.expected_without_trial, PreferredCacheSize(test_case.available, net::GENERATED_BYTE_CODE_CACHE)); // Preferred size for WebUI code cache is not scaled by the trial, and // should never be more than 5 MB. int expected_webui_code_cache_size = std::min(5 * 1024 * 1024, test_case.expected_without_trial); EXPECT_EQ(expected_webui_code_cache_size, PreferredCacheSize(test_case.available, net::GENERATED_WEBUI_BYTE_CODE_CACHE)) << test_case.available; } // Check that the cache size cap is 50% higher for native code caches but is // not scaled for the experiment. EXPECT_EQ(((320 * 1024 * 1024) / 2) * 3, PreferredCacheSize(50000 * 1024 * 1024LL, net::GENERATED_NATIVE_CODE_CACHE)); } // Check no "percent_relative_size" matches default behavior. { base::test::ScopedFeatureList scoped_feature_list; scoped_feature_list.InitAndEnableFeature( disk_cache::kChangeDiskCacheSizeExperiment); for (const auto& test_case : kTestCases) { EXPECT_EQ(test_case.expected_without_trial, PreferredCacheSize(test_case.available)); } // Check that the cache size cap is 50% higher for native code caches. EXPECT_EQ(((320 * 1024 * 1024) / 2) * 3, PreferredCacheSize(50000 * 1024 * 1024LL, net::GENERATED_NATIVE_CODE_CACHE)); } } } // namespace disk_cache