// Copyright (c) 2011 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 "base/file_util.h" #include "base/memory/ref_counted.h" #include "base/message_loop.h" #include "base/path_service.h" #include "base/scoped_temp_dir.h" #include "base/scoped_temp_dir.h" #include "base/string_util.h" #include "base/values.h" #include "chrome/browser/extensions/sandboxed_extension_unpacker.h" #include "chrome/common/chrome_paths.h" #include "chrome/common/extensions/extension.h" #include "chrome/common/extensions/extension_constants.h" #include "chrome/common/extensions/extension_unpacker.h" #include "content/test/test_browser_thread.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" #include "third_party/skia/include/core/SkBitmap.h" namespace errors = extension_manifest_errors; namespace keys = extension_manifest_keys; using content::BrowserThread; using testing::_; using testing::Invoke; namespace { void OnUnpackSuccess(const FilePath& temp_dir, const FilePath& extension_root, const DictionaryValue* original_manifest, const Extension* extension) { // Don't delete temp_dir here, we need to do some post op checking. } } // namespace class MockSandboxedExtensionUnpackerClient : public SandboxedExtensionUnpackerClient { public: virtual ~MockSandboxedExtensionUnpackerClient() {} MOCK_METHOD4(OnUnpackSuccess, void(const FilePath& temp_dir, const FilePath& extension_root, const DictionaryValue* original_manifest, const Extension* extension)); MOCK_METHOD1(OnUnpackFailure, void(const string16& error)); void DelegateToFake() { ON_CALL(*this, OnUnpackSuccess(_, _, _, _)) .WillByDefault(Invoke(::OnUnpackSuccess)); } }; class SandboxedExtensionUnpackerTest : public testing::Test { public: virtual void SetUp() { ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); file_thread_.reset(new content::TestBrowserThread(BrowserThread::FILE, &loop_)); // It will delete itself. client_ = new MockSandboxedExtensionUnpackerClient; client_->DelegateToFake(); } virtual void TearDown() { // Need to destruct SandboxedExtensionUnpacker before the message loop since // it posts a task to it. sandboxed_unpacker_ = NULL; loop_.RunAllPending(); } void SetupUnpacker(const std::string& crx_name) { FilePath original_path; ASSERT_TRUE(PathService::Get(chrome::DIR_TEST_DATA, &original_path)); original_path = original_path.AppendASCII("extensions") .AppendASCII("unpacker") .AppendASCII(crx_name); ASSERT_TRUE(file_util::PathExists(original_path)) << original_path.value(); // Try bots won't let us write into DIR_TEST_DATA, so we have to write the // CRX to the temp directory, and create a subdirectory into which to // unpack it. FilePath crx_path = temp_dir_.path().AppendASCII(crx_name); ASSERT_TRUE(file_util::CopyFile(original_path, crx_path)) << "Original path: " << original_path.value() << ", Crx path: " << crx_path.value(); unpacker_.reset(new ExtensionUnpacker( crx_path, Extension::INTERNAL, Extension::NO_FLAGS)); // Build a temp area where the extension will be unpacked. temp_path_ = temp_dir_.path().AppendASCII("sandboxed_extension_unpacker_test_Temp"); ASSERT_TRUE(file_util::CreateDirectory(temp_path_)); sandboxed_unpacker_ = new SandboxedExtensionUnpacker(crx_path, NULL, Extension::INTERNAL, Extension::NO_FLAGS, client_); // Hack since SandboxedExtensionUnpacker gets its background thread id from // the Start call, but we don't call it here. sandboxed_unpacker_->thread_identifier_ = BrowserThread::FILE; EXPECT_TRUE(PrepareUnpackerEnv()); } bool PrepareUnpackerEnv() { sandboxed_unpacker_->extension_root_ = temp_dir_.path().AppendASCII(extension_filenames::kTempExtensionName); if (!sandboxed_unpacker_->temp_dir_.Set(temp_dir_.path())) return false; sandboxed_unpacker_->public_key_ = "ocnapchkplbmjmpfehjocmjnipfmogkh"; return true; } void OnUnpackSucceeded() { sandboxed_unpacker_->OnUnpackExtensionSucceeded( *unpacker_->parsed_manifest()); } FilePath GetInstallPath() { return temp_dir_.path().AppendASCII( extension_filenames::kTempExtensionName); } bool TempFilesRemoved() { // Check that temporary files were cleaned up. file_util::FileEnumerator::FileType files_and_dirs = static_cast( file_util::FileEnumerator::DIRECTORIES | file_util::FileEnumerator::FILES); file_util::FileEnumerator temp_iterator( temp_path_, true, // recursive files_and_dirs ); int items_not_removed = 0; FilePath item_in_temp; item_in_temp = temp_iterator.Next(); while (!item_in_temp.value().empty()) { items_not_removed++; EXPECT_STREQ(FILE_PATH_LITERAL(""), item_in_temp.value().c_str()) << "File was not removed on success."; item_in_temp = temp_iterator.Next(); } return (items_not_removed == 0); } protected: ScopedTempDir temp_dir_; FilePath temp_path_; MockSandboxedExtensionUnpackerClient* client_; scoped_ptr unpacker_; scoped_refptr sandboxed_unpacker_; MessageLoop loop_; scoped_ptr file_thread_; }; TEST_F(SandboxedExtensionUnpackerTest, NoCatalogsSuccess) { EXPECT_CALL(*client_, OnUnpackSuccess(_, _, _, _)); EXPECT_CALL(*client_, OnUnpackFailure(_)).Times(0); SetupUnpacker("no_l10n.crx"); ASSERT_TRUE(unpacker_->Run()); ASSERT_TRUE(unpacker_->DumpImagesToFile()); ASSERT_TRUE(unpacker_->DumpMessageCatalogsToFile()); // Check that there is no _locales folder. FilePath install_path = GetInstallPath().Append(Extension::kLocaleFolder); EXPECT_FALSE(file_util::PathExists(install_path)); OnUnpackSucceeded(); // Check that there still is no _locales folder. EXPECT_FALSE(file_util::PathExists(install_path)); ASSERT_TRUE(TempFilesRemoved()); } TEST_F(SandboxedExtensionUnpackerTest, WithCatalogsSuccess) { EXPECT_CALL(*client_, OnUnpackSuccess(_, _, _, _)); EXPECT_CALL(*client_, OnUnpackFailure(_)).Times(0); SetupUnpacker("good_l10n.crx"); ASSERT_TRUE(unpacker_->Run()); ASSERT_TRUE(unpacker_->DumpImagesToFile()); ASSERT_TRUE(unpacker_->DumpMessageCatalogsToFile()); // Set timestamp on _locales/en_US/messages.json into the past. FilePath messages_file; messages_file = GetInstallPath().Append(Extension::kLocaleFolder) .AppendASCII("en_US") .Append(Extension::kMessagesFilename); base::PlatformFileInfo old_info; EXPECT_TRUE(file_util::GetFileInfo(messages_file, &old_info)); base::Time old_time = old_info.last_modified - base::TimeDelta::FromSeconds(2); EXPECT_TRUE(file_util::SetLastModifiedTime(messages_file, old_time)); // Refresh old_info, just to be sure. EXPECT_TRUE(file_util::GetFileInfo(messages_file, &old_info)); OnUnpackSucceeded(); // Check that there is newer _locales/en_US/messages.json file. base::PlatformFileInfo new_info; EXPECT_TRUE(file_util::GetFileInfo(messages_file, &new_info)); EXPECT_TRUE(new_info.last_modified > old_info.last_modified); ASSERT_TRUE(TempFilesRemoved()); }