summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--chrome/browser/extensions/extension_file_util.cc342
-rw-r--r--chrome/browser/extensions/extension_file_util.h83
-rw-r--r--chrome/browser/extensions/extension_file_util_unittest.cc167
-rw-r--r--chrome/browser/extensions/extensions_service.cc478
-rw-r--r--chrome/browser/extensions/extensions_service.h69
-rw-r--r--chrome/chrome.gyp3
-rw-r--r--chrome/common/extensions/extension.h1
7 files changed, 651 insertions, 492 deletions
diff --git a/chrome/browser/extensions/extension_file_util.cc b/chrome/browser/extensions/extension_file_util.cc
new file mode 100644
index 0000000..15e3cf8
--- /dev/null
+++ b/chrome/browser/extensions/extension_file_util.cc
@@ -0,0 +1,342 @@
+// Copyright (c) 2009 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 "chrome/browser/extensions/extension_file_util.h"
+
+#include "base/file_util.h"
+#include "base/logging.h"
+#include "base/scoped_temp_dir.h"
+#include "base/string_util.h"
+#include "chrome/common/extensions/extension.h"
+#include "chrome/common/extensions/extension_constants.h"
+#include "chrome/common/json_value_serializer.h"
+#include "net/base/file_stream.h"
+
+namespace extension_file_util {
+
+const char kInstallDirectoryName[] = "Extensions";
+const char kCurrentVersionFileName[] = "Current Version";
+
+bool MoveDirSafely(const FilePath& source_dir, const FilePath& dest_dir) {
+ if (file_util::PathExists(dest_dir)) {
+ if (!file_util::Delete(dest_dir, true))
+ return false;
+ } else {
+ FilePath parent = dest_dir.DirName();
+ if (!file_util::DirectoryExists(parent)) {
+ if (!file_util::CreateDirectory(parent))
+ return false;
+ }
+ }
+
+ if (!file_util::Move(source_dir, dest_dir))
+ return false;
+
+ return true;
+}
+
+bool SetCurrentVersion(const FilePath& dest_dir, const std::string& version,
+ std::string* error) {
+ // Write out the new CurrentVersion file.
+ // <profile>/Extension/<name>/CurrentVersion
+ FilePath current_version = dest_dir.AppendASCII(kCurrentVersionFileName);
+ FilePath current_version_old =
+ current_version.InsertBeforeExtension(FILE_PATH_LITERAL("_old"));
+ if (file_util::PathExists(current_version_old)) {
+ if (!file_util::Delete(current_version_old, false)) {
+ *error = "Couldn't remove CurrentVersion_old file.";
+ return false;
+ }
+ }
+
+ if (file_util::PathExists(current_version)) {
+ if (!file_util::Move(current_version, current_version_old)) {
+ *error = "Couldn't move CurrentVersion file.";
+ return false;
+ }
+ }
+ net::FileStream stream;
+ int flags = base::PLATFORM_FILE_CREATE_ALWAYS | base::PLATFORM_FILE_WRITE;
+ if (stream.Open(current_version, flags) != 0)
+ return false;
+ if (stream.Write(version.c_str(), version.size(), NULL) < 0) {
+ // Restore the old CurrentVersion.
+ if (file_util::PathExists(current_version_old)) {
+ if (!file_util::Move(current_version_old, current_version)) {
+ LOG(WARNING) << "couldn't restore " << current_version_old.value() <<
+ " to " << current_version.value();
+
+ // TODO(erikkay): This is an ugly state to be in. Try harder?
+ }
+ }
+ *error = "Couldn't create CurrentVersion file.";
+ return false;
+ }
+ return true;
+}
+
+bool ReadCurrentVersion(const FilePath& dir, std::string* version_string) {
+ FilePath current_version = dir.AppendASCII(kCurrentVersionFileName);
+ if (file_util::PathExists(current_version)) {
+ if (file_util::ReadFileToString(current_version, version_string)) {
+ TrimWhitespace(*version_string, TRIM_ALL, version_string);
+ return true;
+ }
+ }
+ return false;
+}
+
+Extension::InstallType CompareToInstalledVersion(
+ const FilePath& install_directory, const std::string& id,
+ const std::string& new_version_str, std::string *current_version_str) {
+ CHECK(current_version_str);
+ FilePath dir(install_directory.AppendASCII(id.c_str()));
+ if (!ReadCurrentVersion(dir, current_version_str))
+ return Extension::NEW_INSTALL;
+
+ scoped_ptr<Version> current_version(
+ Version::GetVersionFromString(*current_version_str));
+ scoped_ptr<Version> new_version(
+ Version::GetVersionFromString(new_version_str));
+ int comp = new_version->CompareTo(*current_version);
+ if (comp > 0)
+ return Extension::UPGRADE;
+ else if (comp == 0)
+ return Extension::REINSTALL;
+ else
+ return Extension::DOWNGRADE;
+}
+
+bool SanityCheckExtension(const FilePath& dir) {
+ // Verify that the directory actually exists.
+ // TODO(erikkay): A further step would be to verify that the extension
+ // has actually loaded successfully.
+ FilePath manifest_file(dir.AppendASCII(Extension::kManifestFilename));
+ return file_util::PathExists(dir) && file_util::PathExists(manifest_file);
+}
+
+bool InstallExtension(const FilePath& src_dir,
+ const FilePath& extensions_dir,
+ const std::string& extension_id,
+ const std::string& extension_version,
+ FilePath* version_dir,
+ Extension::InstallType* install_type,
+ std::string* error) {
+ FilePath dest_dir = extensions_dir.AppendASCII(extension_id);
+ *version_dir = dest_dir.AppendASCII(extension_version);
+
+ std::string current_version;
+ *install_type = CompareToInstalledVersion(
+ extensions_dir, extension_id, extension_version, &current_version);
+
+ // Do not allow downgrade.
+ if (*install_type == Extension::DOWNGRADE)
+ return true;
+
+ if (*install_type == Extension::REINSTALL) {
+ if (!SanityCheckExtension(*version_dir)) {
+ // Treat corrupted existing installation as new install case.
+ *install_type = Extension::NEW_INSTALL;
+ } else {
+ return true;
+ }
+ }
+
+ // If anything fails after this, we want to delete the extension dir.
+ ScopedTempDir scoped_version_dir;
+ scoped_version_dir.Set(*version_dir);
+
+ if (!MoveDirSafely(src_dir, *version_dir)) {
+ *error = "Could not move extension directory into profile.";
+ return false;
+ }
+
+ if (!SetCurrentVersion(dest_dir, extension_version, error))
+ return false;
+
+ scoped_version_dir.Take();
+ return true;
+}
+
+Extension* LoadExtension(const FilePath& extension_path, bool require_key,
+ std::string* error) {
+ FilePath manifest_path =
+ extension_path.AppendASCII(Extension::kManifestFilename);
+ if (!file_util::PathExists(manifest_path)) {
+ *error = extension_manifest_errors::kInvalidManifest;
+ return NULL;
+ }
+
+ JSONFileValueSerializer serializer(manifest_path);
+ scoped_ptr<Value> root(serializer.Deserialize(error));
+ if (!root.get())
+ return NULL;
+
+ if (!root->IsType(Value::TYPE_DICTIONARY)) {
+ *error = extension_manifest_errors::kInvalidManifest;
+ return NULL;
+ }
+
+ scoped_ptr<Extension> extension(new Extension(extension_path));
+ if (!extension->InitFromValue(*static_cast<DictionaryValue*>(root.get()),
+ require_key, error))
+ return NULL;
+
+ // Validate icons exist.
+ for (std::map<int, std::string>::const_iterator iter =
+ extension->icons().begin(); iter != extension->icons().end(); ++iter) {
+ if (!file_util::PathExists(extension->GetResourcePath(iter->second))) {
+ *error = StringPrintf("Could not load extension icon '%s'.",
+ iter->second.c_str());
+ return NULL;
+ }
+ }
+
+ // Theme resource validation.
+ if (extension->IsTheme()) {
+ DictionaryValue* images_value = extension->GetThemeImages();
+ DictionaryValue::key_iterator iter = images_value->begin_keys();
+ if (images_value) {
+ while (iter != images_value->end_keys()) {
+ std::string val;
+ if (images_value->GetString(*iter , &val)) {
+ FilePath image_path = extension->path().AppendASCII(val);
+ if (!file_util::PathExists(image_path)) {
+ *error = StringPrintf(
+ "Could not load '%s' for theme.",
+ WideToUTF8(image_path.ToWStringHack()).c_str());
+ return NULL;
+ }
+ }
+ ++iter;
+ }
+ }
+
+ // Themes cannot contain other extension types.
+ return extension.release();
+ }
+
+ // Validate that claimed script resources actually exist.
+ for (size_t i = 0; i < extension->content_scripts().size(); ++i) {
+ const UserScript& script = extension->content_scripts()[i];
+
+ for (size_t j = 0; j < script.js_scripts().size(); j++) {
+ const FilePath& path = script.js_scripts()[j].path();
+ if (!file_util::PathExists(path)) {
+ *error = StringPrintf("Could not load '%s' for content script.",
+ WideToUTF8(path.ToWStringHack()).c_str());
+ return NULL;
+ }
+ }
+
+ for (size_t j = 0; j < script.css_scripts().size(); j++) {
+ const FilePath& path = script.css_scripts()[j].path();
+ if (!file_util::PathExists(path)) {
+ *error = StringPrintf("Could not load '%s' for content script.",
+ WideToUTF8(path.ToWStringHack()).c_str());
+ return NULL;
+ }
+ }
+ }
+
+ for (size_t i = 0; i < extension->plugins().size(); ++i) {
+ const Extension::PluginInfo& plugin = extension->plugins()[i];
+ if (!file_util::PathExists(plugin.path)) {
+ *error = StringPrintf("Could not load '%s' for plugin.",
+ WideToUTF8(plugin.path.ToWStringHack()).c_str());
+ return NULL;
+ }
+ }
+
+ // Validate icon location for page actions.
+ const PageActionMap& page_actions = extension->page_actions();
+ for (PageActionMap::const_iterator i(page_actions.begin());
+ i != page_actions.end(); ++i) {
+ PageAction* page_action = i->second;
+ const std::vector<FilePath>& icon_paths = page_action->icon_paths();
+ for (std::vector<FilePath>::const_iterator iter = icon_paths.begin();
+ iter != icon_paths.end(); ++iter) {
+ FilePath path = *iter;
+ if (!file_util::PathExists(path)) {
+ *error = StringPrintf("Could not load icon '%s' for page action.",
+ WideToUTF8(path.ToWStringHack()).c_str());
+ return NULL;
+ }
+ }
+ }
+
+ return extension.release();
+}
+
+void UninstallExtension(const std::string& id, const FilePath& extensions_dir) {
+ // First, delete the Current Version file. If the directory delete fails, then
+ // at least the extension won't be loaded again.
+ FilePath extension_root = extensions_dir.AppendASCII(id);
+
+ if (!file_util::PathExists(extension_root)) {
+ LOG(WARNING) << "Asked to remove a non-existent extension " << id;
+ return;
+ }
+
+ FilePath current_version_file = extension_root.AppendASCII(
+ kCurrentVersionFileName);
+ if (!file_util::PathExists(current_version_file)) {
+ LOG(WARNING) << "Extension " << id
+ << " does not have a Current Version file.";
+ } else {
+ if (!file_util::Delete(current_version_file, false)) {
+ LOG(WARNING) << "Could not delete Current Version file for extension "
+ << id;
+ return;
+ }
+ }
+
+ // OK, now try and delete the entire rest of the directory. One major place
+ // this can fail is if the extension contains a plugin (stupid plugins). It's
+ // not a big deal though, because we'll notice next time we startup that the
+ // Current Version file is gone and finish the delete then.
+ if (!file_util::Delete(extension_root, true))
+ LOG(WARNING) << "Could not delete directory for extension " << id;
+}
+
+void GarbageCollectExtensions(const FilePath& install_directory) {
+ // Nothing to clean up if it doesn't exist.
+ if (!file_util::DirectoryExists(install_directory))
+ return;
+
+ LOG(INFO) << "Loading installed extensions...";
+ file_util::FileEnumerator enumerator(install_directory,
+ false, // Not recursive.
+ file_util::FileEnumerator::DIRECTORIES);
+ FilePath extension_path;
+ for (extension_path = enumerator.Next(); !extension_path.value().empty();
+ extension_path = enumerator.Next()) {
+ std::string extension_id = WideToASCII(
+ extension_path.BaseName().ToWStringHack());
+
+ // If there is no Current Version file, just delete the directory and move
+ // on. This can legitimately happen when an uninstall does not complete, for
+ // example, when a plugin is in use at uninstall time.
+ FilePath current_version_path = extension_path.AppendASCII(
+ kCurrentVersionFileName);
+ if (!file_util::PathExists(current_version_path)) {
+ LOG(INFO) << "Deleting incomplete install for directory "
+ << WideToASCII(extension_path.ToWStringHack()) << ".";
+ file_util::Delete(extension_path, true); // Recursive.
+ continue;
+ }
+
+ // Ignore directories that aren't valid IDs.
+ if (!Extension::IdIsValid(extension_id)) {
+ LOG(WARNING) << "Invalid extension ID encountered in extensions "
+ "directory: " << extension_id;
+ LOG(INFO) << "Deleting invalid extension directory "
+ << WideToASCII(extension_path.ToWStringHack()) << ".";
+ file_util::Delete(extension_path, true); // Recursive.
+ continue;
+ }
+ }
+}
+
+} // extensionfile_util
diff --git a/chrome/browser/extensions/extension_file_util.h b/chrome/browser/extensions/extension_file_util.h
new file mode 100644
index 0000000..45d0b30
--- /dev/null
+++ b/chrome/browser/extensions/extension_file_util.h
@@ -0,0 +1,83 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_BROWSER_EXTENSIONS_EXTENSION_FILE_UTIL_H_
+#define CHROME_BROWSER_EXTENSIONS_EXTENSION_FILE_UTIL_H_
+
+#include <string>
+
+#include "base/file_path.h"
+#include "chrome/common/extensions/extension.h"
+
+// Utilties for manipulating the on-disk storage of extensions.
+namespace extension_file_util {
+
+// The name of the directory inside the profile that we store installed
+// extension in.
+extern const char kInstallDirectoryName[];
+
+// The name of the file that contains the current version of an installed
+// extension.
+extern const char kCurrentVersionFileName[];
+
+// Move source_dir to dest_dir (it will actually be named dest_dir, not inside
+// dest_dir). If the parent path doesn't exixt, create it. If something else is
+// already there, remove it.
+bool MoveDirSafely(const FilePath& source_dir, const FilePath& dest_dir);
+
+// Updates the Current Version file inside the installed extension.
+bool SetCurrentVersion(const FilePath& dest_dir, const std::string& version,
+ std::string* error);
+
+// Reads the Current Version file.
+bool ReadCurrentVersion(const FilePath& dir, std::string* version_string);
+
+// Determine what type of install (new, upgrade, overinstall, downgrade) a given
+// id/version combination is. Also returns the current version, if any, in
+// current_version_str.
+Extension::InstallType CompareToInstalledVersion(
+ const FilePath& install_directory, const std::string& id,
+ const std::string& new_version_str, std::string *current_version_str);
+
+// Sanity check that the directory has the minimum files to be a working
+// extension.
+bool SanityCheckExtension(const FilePath& extension_root);
+
+// Installs an extension unpacked in |src_dir|. |extensions_dir| is the root
+// directory containing all extensions in the user profile. |extension_id| and
+// |extension_version| are the id and version of the extension contained in
+// |src_dir|.
+//
+// Returns the full path to the destination version directory and the type of
+// install that was attempted.
+//
+// On failure, also returns an error message.
+//
+// NOTE: |src_dir| is not actually copied in the case of downgrades or
+// overinstall of previous verisons of the extension. In that case, the function
+// returns true and install_type is populated.
+bool InstallExtension(const FilePath& src_dir,
+ const FilePath& extensions_dir,
+ const std::string& extension_id,
+ const std::string& extension_version,
+ FilePath* version_dir,
+ Extension::InstallType* install_type,
+ std::string* error);
+
+// Load an extension from the specified directory. Returns NULL on failure, with
+// a description of the error in |error|.
+Extension* LoadExtension(const FilePath& extension_root, bool require_key,
+ std::string* error);
+
+// Uninstalls the extension |id| from the install directory |extensions_dir|.
+void UninstallExtension(const std::string& id, const FilePath& extensions_dir);
+
+// Clean up directories that aren't valid extensions from the install directory.
+// TODO(aa): Also consider passing in a list of known current extensions and
+// removing others?
+void GarbageCollectExtensions(const FilePath& extensions_dir);
+
+} // extension_file_util
+
+#endif // CHROME_BROWSER_EXTENSIONS_EXTENSION_FILE_UTIL_H_
diff --git a/chrome/browser/extensions/extension_file_util_unittest.cc b/chrome/browser/extensions/extension_file_util_unittest.cc
new file mode 100644
index 0000000..6e9b2dc
--- /dev/null
+++ b/chrome/browser/extensions/extension_file_util_unittest.cc
@@ -0,0 +1,167 @@
+// Copyright (c) 2009 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 "chrome/browser/extensions/extension_file_util.h"
+
+#include "base/file_util.h"
+#include "base/scoped_temp_dir.h"
+#include "base/path_service.h"
+#include "chrome/common/chrome_paths.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+TEST(ExtensionFileUtil, MoveDirSafely) {
+ // Create a test directory structure with some data in it.
+ ScopedTempDir temp;
+ ASSERT_TRUE(temp.CreateUniqueTempDir());
+
+ FilePath src_path = temp.path().AppendASCII("src");
+ ASSERT_TRUE(file_util::CreateDirectory(src_path));
+
+ std::string data = "foobar";
+ ASSERT_TRUE(file_util::WriteFile(src_path.AppendASCII("data"),
+ data.c_str(), data.length()));
+
+ // Move it to a path that doesn't exist yet.
+ FilePath dest_path = temp.path().AppendASCII("dest").AppendASCII("dest");
+ ASSERT_TRUE(extension_file_util::MoveDirSafely(src_path, dest_path));
+
+ // The path should get created.
+ ASSERT_TRUE(file_util::DirectoryExists(dest_path));
+
+ // The data should match.
+ std::string data_out;
+ ASSERT_TRUE(file_util::ReadFileToString(dest_path.AppendASCII("data"),
+ &data_out));
+ ASSERT_EQ(data, data_out);
+
+ // The src path should be gone.
+ ASSERT_FALSE(file_util::PathExists(src_path));
+
+ // Create some new test data.
+ ASSERT_TRUE(file_util::CopyDirectory(dest_path, src_path,
+ true)); // recursive
+ data = "hotdog";
+ ASSERT_TRUE(file_util::WriteFile(src_path.AppendASCII("data"),
+ data.c_str(), data.length()));
+
+ // Test again, overwriting the old path.
+ ASSERT_TRUE(extension_file_util::MoveDirSafely(src_path, dest_path));
+ ASSERT_TRUE(file_util::DirectoryExists(dest_path));
+
+ data_out.clear();
+ ASSERT_TRUE(file_util::ReadFileToString(dest_path.AppendASCII("data"),
+ &data_out));
+ ASSERT_EQ(data, data_out);
+ ASSERT_FALSE(file_util::PathExists(src_path));
+}
+
+TEST(ExtensionFileUtil, SetCurrentVersion) {
+ // Create an empty test directory.
+ ScopedTempDir temp;
+ ASSERT_TRUE(temp.CreateUniqueTempDir());
+
+ // Set its version.
+ std::string error;
+ std::string version = "1.0";
+ ASSERT_TRUE(extension_file_util::SetCurrentVersion(temp.path(), version,
+ &error));
+
+ // Test the result.
+ std::string version_out;
+ ASSERT_TRUE(file_util::ReadFileToString(
+ temp.path().AppendASCII(extension_file_util::kCurrentVersionFileName),
+ &version_out));
+ ASSERT_EQ(version, version_out);
+
+ // Try again, overwriting the old value.
+ version = "2.0";
+ version_out.clear();
+ ASSERT_TRUE(extension_file_util::SetCurrentVersion(temp.path(), version,
+ &error));
+ ASSERT_TRUE(file_util::ReadFileToString(
+ temp.path().AppendASCII(extension_file_util::kCurrentVersionFileName),
+ &version_out));
+ ASSERT_EQ(version, version_out);
+}
+
+TEST(ExtensionFileUtil, ReadCurrentVersion) {
+ // Read the version from a valid extension.
+ FilePath extension_path;
+ ASSERT_TRUE(PathService::Get(chrome::DIR_TEST_DATA, &extension_path));
+ extension_path = extension_path.AppendASCII("extensions")
+ .AppendASCII("good")
+ .AppendASCII("Extensions")
+ .AppendASCII("behllobkkfkfnphdnhnkndlbkcpglgmj");
+
+ std::string version;
+ ASSERT_TRUE(extension_file_util::ReadCurrentVersion(extension_path,
+ &version));
+ ASSERT_EQ("1.0.0.0", version);
+
+ // Create an invalid extension and read its current version.
+ ScopedTempDir temp;
+ ASSERT_TRUE(temp.CreateUniqueTempDir());
+ ASSERT_FALSE(extension_file_util::ReadCurrentVersion(temp.path(), &version));
+}
+
+TEST(ExtensionFileUtil, CompareToInstalledVersion) {
+ // Compare to an existing extension.
+ FilePath install_directory;
+ ASSERT_TRUE(PathService::Get(chrome::DIR_TEST_DATA, &install_directory));
+ install_directory = install_directory.AppendASCII("extensions")
+ .AppendASCII("good")
+ .AppendASCII("Extensions");
+
+ std::string id = "behllobkkfkfnphdnhnkndlbkcpglgmj";
+ std::string version = "1.0.0.0";
+ std::string version_out;
+
+ ASSERT_EQ(Extension::UPGRADE, extension_file_util::CompareToInstalledVersion(
+ install_directory, id, "1.0.0.1", &version_out));
+ ASSERT_EQ(version, version_out);
+
+ version_out.clear();
+ ASSERT_EQ(Extension::REINSTALL,
+ extension_file_util::CompareToInstalledVersion(
+ install_directory, id, "1.0.0.0", &version_out));
+ ASSERT_EQ(version, version_out);
+
+ version_out.clear();
+ ASSERT_EQ(Extension::REINSTALL,
+ extension_file_util::CompareToInstalledVersion(
+ install_directory, id, "1.0.0", &version_out));
+ ASSERT_EQ(version, version_out);
+
+ version_out.clear();
+ ASSERT_EQ(Extension::DOWNGRADE,
+ extension_file_util::CompareToInstalledVersion(
+ install_directory, id, "0.0.1.0", &version_out));
+ ASSERT_EQ(version, version_out);
+
+ // Compare to an extension that is missing its Current Version file.
+ ScopedTempDir temp;
+ ASSERT_TRUE(temp.CreateUniqueTempDir());
+ FilePath src = install_directory.AppendASCII(id).AppendASCII(version);
+ FilePath dest = temp.path().AppendASCII(id).AppendASCII(version);
+ ASSERT_TRUE(file_util::CreateDirectory(dest.DirName()));
+ ASSERT_TRUE(file_util::CopyDirectory(src, dest, true));
+
+ version_out.clear();
+ ASSERT_EQ(Extension::NEW_INSTALL,
+ extension_file_util::CompareToInstalledVersion(
+ temp.path(), id, "0.0.1.0", &version_out));
+ ASSERT_EQ("", version_out);
+
+ // Compare to a completely non-existent extension.
+ version_out.clear();
+ ASSERT_EQ(Extension::NEW_INSTALL,
+ extension_file_util::CompareToInstalledVersion(
+ temp.path(), "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx", "0.0.1.0",
+ &version_out));
+ ASSERT_EQ("", version_out);
+}
+
+// TODO(aa): More tests as motivation allows. Maybe steal some from
+// ExtensionsService? Many of them could probably be tested here without the
+// MessageLoop shenanigans.
diff --git a/chrome/browser/extensions/extensions_service.cc b/chrome/browser/extensions/extensions_service.cc
index 1b993e6..0733380 100644
--- a/chrome/browser/extensions/extensions_service.cc
+++ b/chrome/browser/extensions/extensions_service.cc
@@ -7,16 +7,14 @@
#include "app/l10n_util.h"
#include "base/command_line.h"
#include "base/file_util.h"
-#include "base/scoped_temp_dir.h"
-#include "base/stl_util-inl.h"
#include "base/string_util.h"
#include "base/values.h"
-#include "net/base/file_stream.h"
#include "chrome/browser/browser.h"
#include "chrome/browser/browser_list.h"
#include "chrome/browser/browser_process.h"
#include "chrome/browser/chrome_thread.h"
#include "chrome/browser/extensions/extension_browser_event_router.h"
+#include "chrome/browser/extensions/extension_file_util.h"
#include "chrome/browser/extensions/extension_process_manager.h"
#include "chrome/browser/extensions/extension_updater.h"
#include "chrome/browser/extensions/external_extension_provider.h"
@@ -26,9 +24,7 @@
#include "chrome/browser/tab_contents/tab_contents.h"
#include "chrome/common/chrome_switches.h"
#include "chrome/common/extensions/extension.h"
-#include "chrome/common/extensions/extension_constants.h"
#include "chrome/common/extensions/extension_error_reporter.h"
-#include "chrome/common/json_value_serializer.h"
#include "chrome/common/notification_service.h"
#include "chrome/common/pref_names.h"
#include "chrome/common/pref_service.h"
@@ -249,8 +245,9 @@ void ExtensionsService::UninstallExtension(const std::string& extension_id,
// Tell the backend to start deleting installed extensions on the file thread.
if (Extension::LOAD != extension->location()) {
- backend_loop_->PostTask(FROM_HERE, NewRunnableMethod(backend_.get(),
- &ExtensionsServiceBackend::UninstallExtension, extension_id));
+ backend_loop_->PostTask(FROM_HERE, NewRunnableFunction(
+ &extension_file_util::UninstallExtension, extension_id,
+ install_directory_));
}
UnloadExtension(extension_id);
@@ -321,9 +318,8 @@ void ExtensionsService::ReloadExtensions() {
}
void ExtensionsService::GarbageCollectExtensions() {
- backend_loop_->PostTask(FROM_HERE, NewRunnableMethod(backend_.get(),
- &ExtensionsServiceBackend::GarbageCollectExtensions,
- scoped_refptr<ExtensionsService>(this)));
+ backend_loop_->PostTask(FROM_HERE, NewRunnableFunction(
+ &extension_file_util::GarbageCollectExtensions, install_directory_));
}
void ExtensionsService::OnLoadedInstalledExtensions() {
@@ -524,55 +520,6 @@ void ExtensionsServiceBackend::LoadInstalledExtensions(
frontend_, &ExtensionsService::OnLoadedInstalledExtensions));
}
-void ExtensionsServiceBackend::GarbageCollectExtensions(
- scoped_refptr<ExtensionsService> frontend) {
- frontend_ = frontend;
- alert_on_error_ = false;
-
- // Nothing to clean up if it doesn't exist.
- if (!file_util::DirectoryExists(install_directory_))
- return;
-
- FilePath install_directory_absolute(install_directory_);
- file_util::AbsolutePath(&install_directory_absolute);
-
- LOG(INFO) << "Loading installed extensions...";
-
- // Find all child directories in the install directory and load their
- // manifests. Post errors and results to the frontend.
- file_util::FileEnumerator enumerator(install_directory_absolute,
- false, // Not recursive.
- file_util::FileEnumerator::DIRECTORIES);
- FilePath extension_path;
- for (extension_path = enumerator.Next(); !extension_path.value().empty();
- extension_path = enumerator.Next()) {
- std::string extension_id = WideToASCII(
- extension_path.BaseName().ToWStringHack());
-
- // If there is no Current Version file, just delete the directory and move
- // on. This can legitimately happen when an uninstall does not complete, for
- // example, when a plugin is in use at uninstall time.
- FilePath current_version_path = extension_path.AppendASCII(
- ExtensionsService::kCurrentVersionFileName);
- if (!file_util::PathExists(current_version_path)) {
- LOG(INFO) << "Deleting incomplete install for directory "
- << WideToASCII(extension_path.ToWStringHack()) << ".";
- file_util::Delete(extension_path, true); // Recursive.
- continue;
- }
-
- // Ignore directories that aren't valid IDs.
- if (!Extension::IdIsValid(extension_id)) {
- LOG(WARNING) << "Invalid extension ID encountered in extensions "
- "directory: " << extension_id;
- // TODO(erikkay) delete these eventually too...
- continue;
- }
-
- // TODO(erikkay) check for extensions that aren't loaded?
- }
-}
-
void ExtensionsServiceBackend::LoadSingleExtension(
const FilePath& path_in, scoped_refptr<ExtensionsService> frontend) {
frontend_ = frontend;
@@ -586,14 +533,21 @@ void ExtensionsServiceBackend::LoadSingleExtension(
LOG(INFO) << "Loading single extension from " <<
WideToASCII(extension_path.BaseName().ToWStringHack());
- Extension* extension = LoadExtension(extension_path,
- Extension::LOAD,
- false); // Don't require id.
- if (extension) {
- ExtensionList* extensions = new ExtensionList;
- extensions->push_back(extension);
- ReportExtensionsLoaded(extensions);
+ std::string error;
+ Extension* extension = extension_file_util::LoadExtension(
+ extension_path,
+ false, // Don't require id
+ &error);
+
+ if (!extension) {
+ ReportExtensionLoadError(extension_path, error);
+ return;
}
+
+ extension->set_location(Extension::LOAD);
+ ExtensionList* extensions = new ExtensionList;
+ extensions->push_back(extension);
+ ReportExtensionsLoaded(extensions);
}
void ExtensionsServiceBackend::LoadInstalledExtension(
@@ -608,150 +562,26 @@ void ExtensionsServiceBackend::LoadInstalledExtension(
return;
}
- Extension* extension =
- LoadExtension(FilePath(path), location, true); // Require id.
+ std::string error;
+ Extension* extension = extension_file_util::LoadExtension(
+ path,
+ true, // Require id
+ &error);
+
+ if (!extension) {
+ ReportExtensionLoadError(path, error);
+ return;
+ }
// TODO(erikkay) now we only report a single extension loaded at a time.
// Perhaps we should change the notifications to remove ExtensionList.
+ extension->set_location(location);
ExtensionList* extensions = new ExtensionList;
if (extension)
extensions->push_back(extension);
ReportExtensionsLoaded(extensions);
}
-DictionaryValue* ExtensionsServiceBackend::ReadManifest(
- const FilePath& manifest_path, std::string* error) {
- JSONFileValueSerializer serializer(manifest_path);
- scoped_ptr<Value> root(serializer.Deserialize(error));
- if (!root.get())
- return NULL;
-
- if (!root->IsType(Value::TYPE_DICTIONARY)) {
- *error = extension_manifest_errors::kInvalidManifest;
- return NULL;
- }
-
- return static_cast<DictionaryValue*>(root.release());
-}
-
-Extension* ExtensionsServiceBackend::LoadExtension(
- const FilePath& extension_path,
- Extension::Location location,
- bool require_id) {
- FilePath manifest_path =
- extension_path.AppendASCII(Extension::kManifestFilename);
- if (!file_util::PathExists(manifest_path)) {
- ReportExtensionLoadError(extension_path,
- extension_manifest_errors::kInvalidManifest);
- return NULL;
- }
-
- std::string error;
- scoped_ptr<DictionaryValue> root(ReadManifest(manifest_path, &error));
- if (!root.get()) {
- ReportExtensionLoadError(extension_path, error);
- return NULL;
- }
-
- scoped_ptr<Extension> extension(new Extension(extension_path));
- if (!extension->InitFromValue(*root.get(), require_id, &error)) {
- ReportExtensionLoadError(extension_path, error);
- return NULL;
- }
-
- extension->set_location(location);
-
- // Validate icons exist.
- for (std::map<int, std::string>::const_iterator iter =
- extension->icons().begin(); iter != extension->icons().end(); ++iter) {
- if (!file_util::PathExists(extension->GetResourcePath(iter->second))) {
- ReportExtensionLoadError(extension_path,
- StringPrintf("Could not load extension icon '%s'.",
- iter->second.c_str()));
- return false;
- }
- }
-
- // Theme resource validation.
- if (extension->IsTheme()) {
- DictionaryValue* images_value = extension->GetThemeImages();
- if (images_value) {
- DictionaryValue::key_iterator iter = images_value->begin_keys();
- while (iter != images_value->end_keys()) {
- std::string val;
- if (images_value->GetString(*iter , &val)) {
- FilePath image_path = extension->path().AppendASCII(val);
- if (!file_util::PathExists(image_path)) {
- ReportExtensionLoadError(extension_path,
- StringPrintf("Could not load '%s' for theme.",
- WideToUTF8(image_path.ToWStringHack()).c_str()));
- return NULL;
- }
- }
- ++iter;
- }
- }
-
- // Themes cannot contain other extension types.
- return extension.release();
- }
-
- // Validate that claimed script resources actually exist.
- for (size_t i = 0; i < extension->content_scripts().size(); ++i) {
- const UserScript& script = extension->content_scripts()[i];
-
- for (size_t j = 0; j < script.js_scripts().size(); j++) {
- const FilePath& path = script.js_scripts()[j].path();
- if (!file_util::PathExists(path)) {
- ReportExtensionLoadError(extension_path,
- StringPrintf("Could not load '%s' for content script.",
- WideToUTF8(path.ToWStringHack()).c_str()));
- return NULL;
- }
- }
-
- for (size_t j = 0; j < script.css_scripts().size(); j++) {
- const FilePath& path = script.css_scripts()[j].path();
- if (!file_util::PathExists(path)) {
- ReportExtensionLoadError(extension_path,
- StringPrintf("Could not load '%s' for content script.",
- WideToUTF8(path.ToWStringHack()).c_str()));
- return NULL;
- }
- }
- }
-
- for (size_t i = 0; i < extension->plugins().size(); ++i) {
- const Extension::PluginInfo& plugin = extension->plugins()[i];
- if (!file_util::PathExists(plugin.path)) {
- ReportExtensionLoadError(extension_path,
- StringPrintf("Could not load '%s' for plugin.",
- WideToUTF8(plugin.path.ToWStringHack()).c_str()));
- return NULL;
- }
- }
-
- // Validate icon location for page actions.
- const PageActionMap& page_actions = extension->page_actions();
- for (PageActionMap::const_iterator i(page_actions.begin());
- i != page_actions.end(); ++i) {
- PageAction* page_action = i->second;
- const std::vector<FilePath>& icon_paths = page_action->icon_paths();
- for (std::vector<FilePath>::const_iterator iter = icon_paths.begin();
- iter != icon_paths.end(); ++iter) {
- FilePath path = *iter;
- if (!file_util::PathExists(path)) {
- ReportExtensionLoadError(extension_path,
- StringPrintf("Could not load icon '%s' for page action.",
- WideToUTF8(path.ToWStringHack()).c_str()));
- return NULL;
- }
- }
- }
-
- return extension.release();
-}
-
void ExtensionsServiceBackend::ReportExtensionLoadError(
const FilePath& extension_path, const std::string &error) {
// TODO(port): note that this isn't guaranteed to work properly on Linux.
@@ -767,123 +597,6 @@ void ExtensionsServiceBackend::ReportExtensionsLoaded(
frontend_, &ExtensionsService::OnExtensionsLoaded, extensions));
}
-bool ExtensionsServiceBackend::ReadCurrentVersion(const FilePath& dir,
- std::string* version_string) {
- FilePath current_version =
- dir.AppendASCII(ExtensionsService::kCurrentVersionFileName);
- if (file_util::PathExists(current_version)) {
- if (file_util::ReadFileToString(current_version, version_string)) {
- TrimWhitespace(*version_string, TRIM_ALL, version_string);
- return true;
- }
- }
- return false;
-}
-
-Extension::InstallType ExtensionsServiceBackend::CompareToInstalledVersion(
- const std::string& id,
- const std::string& new_version_str,
- std::string *current_version_str) {
- CHECK(current_version_str);
- FilePath dir(install_directory_.AppendASCII(id.c_str()));
- if (!ReadCurrentVersion(dir, current_version_str))
- return Extension::NEW_INSTALL;
-
- scoped_ptr<Version> current_version(
- Version::GetVersionFromString(*current_version_str));
- scoped_ptr<Version> new_version(
- Version::GetVersionFromString(new_version_str));
- int comp = new_version->CompareTo(*current_version);
- if (comp > 0)
- return Extension::UPGRADE;
- else if (comp == 0)
- return Extension::REINSTALL;
- else
- return Extension::DOWNGRADE;
-}
-
-bool ExtensionsServiceBackend::NeedsReinstall(const std::string& id,
- const std::string& current_version) {
- // Verify that the directory actually exists.
- // TODO(erikkay): A further step would be to verify that the extension
- // has actually loaded successfully.
- FilePath dir(install_directory_.AppendASCII(id.c_str()));
- FilePath version_dir(dir.AppendASCII(current_version));
- return !file_util::PathExists(version_dir);
-}
-
-bool ExtensionsServiceBackend::InstallDirSafely(const FilePath& source_dir,
- const FilePath& dest_dir) {
- if (file_util::PathExists(dest_dir)) {
- // By the time we get here, it should be safe to assume that this directory
- // is not currently in use (it's not the current active version).
- if (!file_util::Delete(dest_dir, true)) {
- ReportExtensionInstallError(source_dir,
- "Can't delete existing version directory.");
- return false;
- }
- } else {
- FilePath parent = dest_dir.DirName();
- if (!file_util::DirectoryExists(parent)) {
- if (!file_util::CreateDirectory(parent)) {
- ReportExtensionInstallError(source_dir,
- "Couldn't create extension directory.");
- return false;
- }
- }
- }
- if (!file_util::Move(source_dir, dest_dir)) {
- ReportExtensionInstallError(source_dir,
- "Couldn't move temporary directory.");
- return false;
- }
-
- return true;
-}
-
-bool ExtensionsServiceBackend::SetCurrentVersion(const FilePath& dest_dir,
- const std::string& version) {
- // Write out the new CurrentVersion file.
- // <profile>/Extension/<name>/CurrentVersion
- FilePath current_version =
- dest_dir.AppendASCII(ExtensionsService::kCurrentVersionFileName);
- FilePath current_version_old =
- current_version.InsertBeforeExtension(FILE_PATH_LITERAL("_old"));
- if (file_util::PathExists(current_version_old)) {
- if (!file_util::Delete(current_version_old, false)) {
- ReportExtensionInstallError(dest_dir,
- "Couldn't remove CurrentVersion_old file.");
- return false;
- }
- }
- if (file_util::PathExists(current_version)) {
- if (!file_util::Move(current_version, current_version_old)) {
- ReportExtensionInstallError(dest_dir,
- "Couldn't move CurrentVersion file.");
- return false;
- }
- }
- net::FileStream stream;
- int flags = base::PLATFORM_FILE_CREATE_ALWAYS | base::PLATFORM_FILE_WRITE;
- if (stream.Open(current_version, flags) != 0)
- return false;
- if (stream.Write(version.c_str(), version.size(), NULL) < 0) {
- // Restore the old CurrentVersion.
- if (file_util::PathExists(current_version_old)) {
- if (!file_util::Move(current_version_old, current_version)) {
- LOG(WARNING) << "couldn't restore " << current_version_old.value() <<
- " to " << current_version.value();
-
- // TODO(erikkay): This is an ugly state to be in. Try harder?
- }
- }
- ReportExtensionInstallError(dest_dir,
- "Couldn't create CurrentVersion file.");
- return false;
- }
- return true;
-}
-
void ExtensionsServiceBackend::InstallExtension(
const FilePath& extension_path, bool from_gallery,
scoped_refptr<ExtensionsService> frontend) {
@@ -992,53 +705,37 @@ void ExtensionsServiceBackend::OnExtensionUnpacked(
// If an expected id was provided, make sure it matches.
if (!expected_id.empty() && expected_id != extension->id()) {
- std::string error_msg = "ID in new extension manifest (";
- error_msg += extension->id();
- error_msg += ") does not match expected ID (";
- error_msg += expected_id;
- error_msg += ")";
- ReportExtensionInstallError(crx_path, error_msg);
+ ReportExtensionInstallError(crx_path,
+ StringPrintf("ID in new extension manifest (%s) does not match "
+ "expected id (%s)", extension->id().c_str(),
+ expected_id.c_str()));
return;
}
- // <profile>/Extensions/<id>
- FilePath dest_dir = install_directory_.AppendASCII(extension->id());
- std::string version = extension->VersionString();
- std::string current_version;
- Extension::InstallType install_type =
- CompareToInstalledVersion(extension->id(), version, &current_version);
-
- // Do not allow downgrade.
- if (install_type == Extension::DOWNGRADE) {
- ReportExtensionInstallError(crx_path,
- "Error: Attempt to downgrade extension from more recent version.");
+ FilePath version_dir;
+ Extension::InstallType install_type = Extension::INSTALL_ERROR;
+ std::string error_msg;
+ if (!extension_file_util::InstallExtension(unpacked_path, install_directory_,
+ extension->id(),
+ extension->VersionString(),
+ &version_dir,
+ &install_type, &error_msg)) {
+ ReportExtensionInstallError(crx_path, error_msg);
return;
}
- if (install_type == Extension::REINSTALL) {
- if (NeedsReinstall(extension->id(), current_version)) {
- // Treat corrupted existing installation as new install case.
- install_type = Extension::NEW_INSTALL;
- } else {
- // The client may use this as a signal (to switch themes, for instance).
- ReportExtensionOverinstallAttempted(extension->id(), crx_path);
- return;
- }
+ if (install_type == Extension::DOWNGRADE) {
+ ReportExtensionInstallError(crx_path, "Attempted to downgrade extension.");
+ return;
}
- // <profile>/Extensions/<dir_name>/<version>
- FilePath version_dir = dest_dir.AppendASCII(version);
extension->set_path(version_dir);
- // If anything fails after this, we want to delete the extension dir.
- ScopedTempDir scoped_version_dir;
- scoped_version_dir.Set(version_dir);
-
- if (!InstallDirSafely(unpacked_path, version_dir))
- return;
-
- if (!SetCurrentVersion(dest_dir, version))
+ if (install_type == Extension::REINSTALL) {
+ // The client may use this as a signal (to switch themes, for instance).
+ ReportExtensionOverinstallAttempted(extension->id(), crx_path);
return;
+ }
frontend_loop_->PostTask(FROM_HERE, NewRunnableMethod(
frontend_, &ExtensionsService::OnExtensionInstalled, crx_path,
@@ -1051,8 +748,6 @@ void ExtensionsServiceBackend::OnExtensionUnpacked(
// Hand off ownership of the extension to the frontend.
extensions->push_back(extension_deleter.release());
ReportExtensionsLoaded(extensions);
-
- scoped_version_dir.Take();
}
void ExtensionsServiceBackend::ReportExtensionInstallError(
@@ -1078,23 +773,6 @@ void ExtensionsServiceBackend::ReportExtensionOverinstallAttempted(
path));
}
-bool ExtensionsServiceBackend::ShouldSkipInstallingExtension(
- const std::set<std::string>& ids_to_ignore,
- const std::string& id) {
- if (ids_to_ignore.find(id) != ids_to_ignore.end()) {
- LOG(INFO) << "Skipping uninstalled external extension " << id;
- return true;
- }
- return false;
-}
-
-void ExtensionsServiceBackend::CheckVersionAndInstallExtension(
- const std::string& id, const Version* extension_version,
- const FilePath& extension_path) {
- if (ShouldInstall(id, extension_version))
- InstallOrUpdateExtension(FilePath(extension_path), false, id, false);
-}
-
bool ExtensionsServiceBackend::LookupExternalExtension(
const std::string& id, Version** version, Extension::Location* location) {
scoped_ptr<Version> extension_version;
@@ -1154,41 +832,6 @@ bool ExtensionsServiceBackend::CheckExternalUninstall(
return true; // This is not a known extension, uninstall.
}
-// Assumes that the extension isn't currently loaded or in use.
-void ExtensionsServiceBackend::UninstallExtension(
- const std::string& extension_id) {
- // First, delete the Current Version file. If the directory delete fails, then
- // at least the extension won't be loaded again.
- FilePath extension_directory = install_directory_.AppendASCII(extension_id);
-
- if (!file_util::PathExists(extension_directory)) {
- LOG(WARNING) << "Asked to remove a non-existent extension " << extension_id;
- return;
- }
-
- FilePath current_version_file = extension_directory.AppendASCII(
- ExtensionsService::kCurrentVersionFileName);
- if (!file_util::PathExists(current_version_file)) {
- LOG(WARNING) << "Extension " << extension_id
- << " does not have a Current Version file.";
- } else {
- if (!file_util::Delete(current_version_file, false)) {
- LOG(WARNING) << "Could not delete Current Version file for extension "
- << extension_id;
- return;
- }
- }
-
- // OK, now try and delete the entire rest of the directory. One major place
- // this can fail is if the extension contains a plugin (stupid plugins). It's
- // not a big deal though, because we'll notice next time we startup that the
- // Current Version file is gone and finish the delete then.
- if (!file_util::Delete(extension_directory, true)) {
- LOG(WARNING) << "Could not delete directory for extension "
- << extension_id;
- }
-}
-
void ExtensionsServiceBackend::ClearProvidersForTesting() {
external_extension_providers_.clear();
}
@@ -1203,19 +846,8 @@ void ExtensionsServiceBackend::SetProviderForTesting(
void ExtensionsServiceBackend::OnExternalExtensionFound(
const std::string& id, const Version* version, const FilePath& path) {
- CheckVersionAndInstallExtension(id, version, path);
-}
-
-bool ExtensionsServiceBackend::ShouldInstall(const std::string& id,
- const Version* version) {
- std::string current_version;
- Extension::InstallType install_type =
- CompareToInstalledVersion(id, version->GetString(), &current_version);
-
- if (install_type == Extension::DOWNGRADE)
- return false;
-
- return (install_type == Extension::UPGRADE ||
- install_type == Extension::NEW_INSTALL ||
- NeedsReinstall(id, current_version));
+ InstallOrUpdateExtension(path,
+ false, // not from gallery
+ id, // expected id
+ true); // silent
}
diff --git a/chrome/browser/extensions/extensions_service.h b/chrome/browser/extensions/extensions_service.h
index 5fc036b..a62ebee 100644
--- a/chrome/browser/extensions/extensions_service.h
+++ b/chrome/browser/extensions/extensions_service.h
@@ -281,10 +281,6 @@ class ExtensionsServiceBackend
void LoadInstalledExtensions(scoped_refptr<ExtensionsService> frontend,
InstalledExtensions* installed);
- // Scans the extension installation directory to look for partially installed
- // or extensions to uninstall.
- void GarbageCollectExtensions(scoped_refptr<ExtensionsService> frontend);
-
// Loads a single extension from |path| where |path| is the top directory of
// a specific extension where its manifest file lives.
// Errors are reported through ExtensionErrorReporter. On completion,
@@ -314,11 +310,6 @@ class ExtensionsServiceBackend
void CheckForExternalUpdates(std::set<std::string> ids_to_ignore,
scoped_refptr<ExtensionsService> frontend);
- // Deletes all versions of the extension from the filesystem. Note that only
- // extensions whose location() == INTERNAL can be uninstalled. Attempting to
- // uninstall other extensions will silently fail.
- void UninstallExtension(const std::string& extension_id);
-
// Clear all ExternalExtensionProviders.
void ClearProvidersForTesting();
@@ -339,22 +330,6 @@ class ExtensionsServiceBackend
void LoadInstalledExtension(const std::string& id, const FilePath& path,
Extension::Location location);
- // Utility function to read an extension manifest and return it as a
- // DictionaryValue. If it fails, NULL is returned and |error| contains an
- // appropriate message.
- DictionaryValue* ReadManifest(const FilePath& manifest_path,
- std::string* error);
-
- // Load a single extension from |extension_path|, the top directory of
- // a specific extension where its manifest file lives.
- Extension* LoadExtension(const FilePath& extension_path,
- Extension::Location location,
- bool require_id);
-
- // Load a single extension from |extension_path|, the top directory of
- // a versioned extension where its Current Version file lives.
- Extension* LoadExtensionCurrentVersion(const FilePath& extension_path);
-
// Install a crx file at |extension_path|. If |expected_id| is not empty, it's
// verified against the extension's manifest before installation. If the
// extension is already installed, install the new version only if its version
@@ -395,17 +370,6 @@ class ExtensionsServiceBackend
void ReportExtensionOverinstallAttempted(const std::string& id,
const FilePath& path);
- // Checks a set of strings (containing id's to ignore) in order to determine
- // if the extension should be installed.
- bool ShouldSkipInstallingExtension(const std::set<std::string>& ids_to_ignore,
- const std::string& id);
-
- // Installs the extension if the extension is a newer version or if the
- // extension hasn't been installed before.
- void CheckVersionAndInstallExtension(const std::string& id,
- const Version* extension_version,
- const FilePath& extension_path);
-
// Lookup an external extension by |id| by going through all registered
// external extension providers until we find a provider that contains an
// extension that matches. If |version| is not NULL, the extension version
@@ -416,45 +380,12 @@ class ExtensionsServiceBackend
Version** version,
Extension::Location* location);
- // Read the manifest from the front of the extension file.
- // Caller takes ownership of return value.
- DictionaryValue* ReadManifest(const FilePath& extension_path);
-
- // Reads the Current Version file from |dir| into |version_string|.
- bool ReadCurrentVersion(const FilePath& dir, std::string* version_string);
-
- // Look for an existing installation of the extension |id| & return
- // an InstallType that would result from installing |new_version_str|.
- Extension::InstallType CompareToInstalledVersion(const std::string& id,
- const std::string& new_version_str, std::string* current_version_str);
-
- // Does an existing installed extension need to be reinstalled.
- bool NeedsReinstall(const std::string& id,
- const std::string& current_version);
-
- // Install the extension dir by moving it from |source| to |dest| safely.
- bool InstallDirSafely(const FilePath& source,
- const FilePath& dest);
-
- // Update the CurrentVersion file in |dest_dir| to |version|.
- bool SetCurrentVersion(const FilePath& dest_dir,
- const std::string& version);
-
// For the extension in |version_path| with |id|, check to see if it's an
// externally managed extension. If so return true if it should be
// uninstalled.
bool CheckExternalUninstall(const std::string& id,
Extension::Location location);
- // Should an extension of |id| and |version| be installed?
- // Returns true if no extension of type |id| is installed or if |version|
- // is greater than the current installed version.
- bool ShouldInstall(const std::string& id, const Version* version);
-
- // The name of a temporary directory to install an extension into for
- // validation before finalizing install.
- static const char* kTempExtensionName;
-
// This is a naked pointer which is set by each entry point.
// The entry point is responsible for ensuring lifetime.
ExtensionsService* frontend_;
diff --git a/chrome/chrome.gyp b/chrome/chrome.gyp
index f3d1e51..2ffa560 100644
--- a/chrome/chrome.gyp
+++ b/chrome/chrome.gyp
@@ -950,6 +950,8 @@
'browser/extensions/extension_dom_ui.h',
'browser/extensions/extension_event_names.cc',
'browser/extensions/extension_event_names.h',
+ 'browser/extensions/extension_file_util.cc',
+ 'browser/extensions/extension_file_util.h',
'browser/extensions/extension_function.cc',
'browser/extensions/extension_function.h',
'browser/extensions/extension_function_dispatcher.cc',
@@ -3768,6 +3770,7 @@
'browser/download/download_request_manager_unittest.cc',
'browser/download/save_package_unittest.cc',
'browser/encoding_menu_controller_unittest.cc',
+ 'browser/extensions/extension_file_util_unittest.cc',
'browser/extensions/extension_messages_unittest.cc',
'browser/extensions/extension_process_manager_unittest.cc',
'browser/extensions/extension_ui_unittest.cc',
diff --git a/chrome/common/extensions/extension.h b/chrome/common/extensions/extension.h
index 7b2e12e..dc95ef2 100644
--- a/chrome/common/extensions/extension.h
+++ b/chrome/common/extensions/extension.h
@@ -39,6 +39,7 @@ class Extension {
};
enum InstallType {
+ INSTALL_ERROR,
DOWNGRADE,
REINSTALL,
UPGRADE,