// Copyright (c) 2006-2008 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 #include #include #include "base/base_paths.h" #include "base/file_util.h" #include "base/logging.h" #include "base/path_service.h" #include "base/process_util.h" #include "base/scoped_ptr.h" #include "base/string_util.h" #include "chrome/installer/util/delete_tree_work_item.h" #include "chrome/installer/util/work_item.h" #include "testing/gtest/include/gtest/gtest.h" namespace { class DeleteTreeWorkItemTest : public testing::Test { protected: virtual void SetUp() { // Name a subdirectory of the user temp directory. ASSERT_TRUE(PathService::Get(base::DIR_TEMP, &test_dir_)); test_dir_ = test_dir_.AppendASCII("DeleteTreeWorkItemTest"); // Create a fresh, empty copy of this test directory. file_util::Delete(test_dir_, true); file_util::CreateDirectoryW(test_dir_); ASSERT_TRUE(file_util::PathExists(test_dir_)); } virtual void TearDown() { // Clean up test directory ASSERT_TRUE(file_util::Delete(test_dir_, false)); ASSERT_FALSE(file_util::PathExists(test_dir_)); } // the path to temporary directory used to contain the test operations FilePath test_dir_; }; // Simple function to dump some text into a new file. void CreateTextFile(const std::wstring& filename, const std::wstring& contents) { std::ofstream file; file.open(filename.c_str()); ASSERT_TRUE(file.is_open()); file << contents; file.close(); } wchar_t text_content_1[] = L"delete me"; wchar_t text_content_2[] = L"delete me as well"; }; // Delete a tree without key path. Everything should be deleted. TEST_F(DeleteTreeWorkItemTest, DeleteTreeNoKeyPath) { // Create tree to be deleted std::wstring dir_name_delete(test_dir_.ToWStringHack()); file_util::AppendToPath(&dir_name_delete, L"to_be_delete"); CreateDirectory(dir_name_delete.c_str(), NULL); ASSERT_TRUE(file_util::PathExists(dir_name_delete)); std::wstring dir_name_delete_1(dir_name_delete); file_util::AppendToPath(&dir_name_delete_1, L"1"); CreateDirectory(dir_name_delete_1.c_str(), NULL); ASSERT_TRUE(file_util::PathExists(dir_name_delete_1)); std::wstring dir_name_delete_2(dir_name_delete); file_util::AppendToPath(&dir_name_delete_2, L"2"); CreateDirectory(dir_name_delete_2.c_str(), NULL); ASSERT_TRUE(file_util::PathExists(dir_name_delete_2)); std::wstring file_name_delete_1(dir_name_delete_1); file_util::AppendToPath(&file_name_delete_1, L"File_1.txt"); CreateTextFile(file_name_delete_1, text_content_1); ASSERT_TRUE(file_util::PathExists(file_name_delete_1)); std::wstring file_name_delete_2(dir_name_delete_2); file_util::AppendToPath(&file_name_delete_2, L"File_2.txt"); CreateTextFile(file_name_delete_2, text_content_1); ASSERT_TRUE(file_util::PathExists(file_name_delete_2)); // test Do() scoped_ptr work_item( WorkItem::CreateDeleteTreeWorkItem(dir_name_delete, std::wstring())); EXPECT_TRUE(work_item->Do()); // everything should be gone EXPECT_FALSE(file_util::PathExists(file_name_delete_1)); EXPECT_FALSE(file_util::PathExists(file_name_delete_2)); EXPECT_FALSE(file_util::PathExists(dir_name_delete)); work_item->Rollback(); // everything should come back EXPECT_TRUE(file_util::PathExists(file_name_delete_1)); EXPECT_TRUE(file_util::PathExists(file_name_delete_2)); EXPECT_TRUE(file_util::PathExists(dir_name_delete)); } // Delete a tree with keypath but not in use. Everything should be gone. // Rollback should bring back everything TEST_F(DeleteTreeWorkItemTest, DeleteTree) { // Create tree to be deleted std::wstring dir_name_delete(test_dir_.ToWStringHack()); file_util::AppendToPath(&dir_name_delete, L"to_be_delete"); CreateDirectory(dir_name_delete.c_str(), NULL); ASSERT_TRUE(file_util::PathExists(dir_name_delete)); std::wstring dir_name_delete_1(dir_name_delete); file_util::AppendToPath(&dir_name_delete_1, L"1"); CreateDirectory(dir_name_delete_1.c_str(), NULL); ASSERT_TRUE(file_util::PathExists(dir_name_delete_1)); std::wstring dir_name_delete_2(dir_name_delete); file_util::AppendToPath(&dir_name_delete_2, L"2"); CreateDirectory(dir_name_delete_2.c_str(), NULL); ASSERT_TRUE(file_util::PathExists(dir_name_delete_2)); std::wstring file_name_delete_1(dir_name_delete_1); file_util::AppendToPath(&file_name_delete_1, L"File_1.txt"); CreateTextFile(file_name_delete_1, text_content_1); ASSERT_TRUE(file_util::PathExists(file_name_delete_1)); std::wstring file_name_delete_2(dir_name_delete_2); file_util::AppendToPath(&file_name_delete_2, L"File_2.txt"); CreateTextFile(file_name_delete_2, text_content_1); ASSERT_TRUE(file_util::PathExists(file_name_delete_2)); // test Do() scoped_ptr work_item( WorkItem::CreateDeleteTreeWorkItem(dir_name_delete, file_name_delete_1)); EXPECT_TRUE(work_item->Do()); // everything should be gone EXPECT_FALSE(file_util::PathExists(file_name_delete_1)); EXPECT_FALSE(file_util::PathExists(file_name_delete_2)); EXPECT_FALSE(file_util::PathExists(dir_name_delete)); work_item->Rollback(); // everything should come back EXPECT_TRUE(file_util::PathExists(file_name_delete_1)); EXPECT_TRUE(file_util::PathExists(file_name_delete_2)); EXPECT_TRUE(file_util::PathExists(dir_name_delete)); } // Delete a tree with key_path in use. Everything should still be there. TEST_F(DeleteTreeWorkItemTest, DeleteTreeInUse) { // Create tree to be deleted std::wstring dir_name_delete(test_dir_.ToWStringHack()); file_util::AppendToPath(&dir_name_delete, L"to_be_delete"); CreateDirectory(dir_name_delete.c_str(), NULL); ASSERT_TRUE(file_util::PathExists(dir_name_delete)); std::wstring dir_name_delete_1(dir_name_delete); file_util::AppendToPath(&dir_name_delete_1, L"1"); CreateDirectory(dir_name_delete_1.c_str(), NULL); ASSERT_TRUE(file_util::PathExists(dir_name_delete_1)); std::wstring dir_name_delete_2(dir_name_delete); file_util::AppendToPath(&dir_name_delete_2, L"2"); CreateDirectory(dir_name_delete_2.c_str(), NULL); ASSERT_TRUE(file_util::PathExists(dir_name_delete_2)); std::wstring file_name_delete_1(dir_name_delete_1); file_util::AppendToPath(&file_name_delete_1, L"File_1.txt"); CreateTextFile(file_name_delete_1, text_content_1); ASSERT_TRUE(file_util::PathExists(file_name_delete_1)); std::wstring file_name_delete_2(dir_name_delete_2); file_util::AppendToPath(&file_name_delete_2, L"File_2.txt"); CreateTextFile(file_name_delete_2, text_content_1); ASSERT_TRUE(file_util::PathExists(file_name_delete_2)); // Create a key path file. std::wstring key_path(dir_name_delete); file_util::AppendToPath(&key_path, L"key_file.exe"); wchar_t exe_full_path_str[MAX_PATH]; ::GetModuleFileNameW(NULL, exe_full_path_str, MAX_PATH); std::wstring exe_full_path(exe_full_path_str); file_util::CopyFile(exe_full_path, key_path); ASSERT_TRUE(file_util::PathExists(key_path)); LOG(INFO) << "copy ourself from " << exe_full_path << " to " << key_path; // Run the key path file to keep it in use. STARTUPINFOW si = {sizeof(si)}; PROCESS_INFORMATION pi = {0}; ASSERT_TRUE( ::CreateProcessW(NULL, const_cast(key_path.c_str()), NULL, NULL, FALSE, CREATE_NO_WINDOW | CREATE_SUSPENDED, NULL, NULL, &si, &pi)); // test Do(). { scoped_ptr work_item( WorkItem::CreateDeleteTreeWorkItem(dir_name_delete, key_path)); // delete should fail as file in use. EXPECT_FALSE(work_item->Do()); } // verify everything is still there. EXPECT_TRUE(file_util::PathExists(key_path)); EXPECT_TRUE(file_util::PathExists(file_name_delete_1)); EXPECT_TRUE(file_util::PathExists(file_name_delete_2)); TerminateProcess(pi.hProcess, 0); // make sure the handle is closed. WaitForSingleObject(pi.hProcess, INFINITE); }