| // Copyright (c) 2012 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/files/file_enumerator.h" |
| #include "base/files/scoped_temp_dir.h" |
| #include "base/memory/ref_counted.h" |
| #include "base/message_loop.h" |
| #include "base/path_service.h" |
| #include "base/string_util.h" |
| #include "base/values.h" |
| #include "chrome/browser/extensions/sandboxed_unpacker.h" |
| #include "chrome/common/chrome_paths.h" |
| #include "chrome/common/extensions/extension.h" |
| #include "chrome/common/extensions/unpacker.h" |
| #include "content/public/test/test_browser_thread.h" |
| #include "extensions/common/constants.h" |
| #include "testing/gmock/include/gmock/gmock.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "third_party/skia/include/core/SkBitmap.h" |
| |
| using content::BrowserThread; |
| using testing::_; |
| using testing::Invoke; |
| |
| namespace { |
| |
| void OnUnpackSuccess(const base::FilePath& temp_dir, |
| const base::FilePath& extension_root, |
| const DictionaryValue* original_manifest, |
| const extensions::Extension* extension) { |
| // Don't delete temp_dir here, we need to do some post op checking. |
| } |
| |
| } // namespace |
| |
| namespace extensions { |
| |
| class MockSandboxedUnpackerClient : public SandboxedUnpackerClient { |
| public: |
| MOCK_METHOD4(OnUnpackSuccess, |
| void(const base::FilePath& temp_dir, |
| const base::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)); |
| } |
| |
| protected: |
| virtual ~MockSandboxedUnpackerClient() {} |
| }; |
| |
| class SandboxedUnpackerTest : public testing::Test { |
| public: |
| virtual void SetUp() { |
| ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); |
| ASSERT_TRUE(extensions_dir_.CreateUniqueTempDir()); |
| file_thread_.reset(new content::TestBrowserThread(BrowserThread::FILE, |
| &loop_)); |
| // It will delete itself. |
| client_ = new MockSandboxedUnpackerClient; |
| client_->DelegateToFake(); |
| } |
| |
| virtual void TearDown() { |
| // Need to destruct SandboxedUnpacker before the message loop since |
| // it posts a task to it. |
| sandboxed_unpacker_ = NULL; |
| loop_.RunUntilIdle(); |
| } |
| |
| void SetupUnpacker(const std::string& crx_name) { |
| base::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. |
| base::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 Unpacker( |
| crx_path, std::string(), Manifest::INTERNAL, Extension::NO_FLAGS)); |
| |
| // Build a temp area where the extension will be unpacked. |
| temp_path_ = |
| temp_dir_.path().AppendASCII("sandboxed_unpacker_test_Temp"); |
| ASSERT_TRUE(file_util::CreateDirectory(temp_path_)); |
| |
| sandboxed_unpacker_ = |
| new SandboxedUnpacker( |
| crx_path, false, Manifest::INTERNAL, Extension::NO_FLAGS, |
| extensions_dir_.path(), |
| BrowserThread::GetMessageLoopProxyForThread(BrowserThread::FILE), |
| client_); |
| |
| 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()); |
| } |
| |
| base::FilePath GetInstallPath() { |
| return temp_dir_.path().AppendASCII( |
| extension_filenames::kTempExtensionName); |
| } |
| |
| bool TempFilesRemoved() { |
| // Check that temporary files were cleaned up. |
| int files_and_dirs = base::FileEnumerator::DIRECTORIES | |
| base::FileEnumerator::FILES; |
| |
| base::FileEnumerator temp_iterator( |
| temp_path_, |
| true, // recursive |
| files_and_dirs |
| ); |
| int items_not_removed = 0; |
| base::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: |
| base::ScopedTempDir temp_dir_; |
| base::ScopedTempDir extensions_dir_; |
| base::FilePath temp_path_; |
| MockSandboxedUnpackerClient* client_; |
| scoped_ptr<Unpacker> unpacker_; |
| scoped_refptr<SandboxedUnpacker> sandboxed_unpacker_; |
| base::MessageLoop loop_; |
| scoped_ptr<content::TestBrowserThread> file_thread_; |
| }; |
| |
| TEST_F(SandboxedUnpackerTest, 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. |
| base::FilePath install_path = |
| GetInstallPath().Append(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(SandboxedUnpackerTest, 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. |
| base::FilePath messages_file; |
| messages_file = GetInstallPath().Append(kLocaleFolder) |
| .AppendASCII("en_US") |
| .Append(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()); |
| } |
| |
| } // namespace extensions |