| Index: chrome/common/extensions/extension_file_util.cc
|
| diff --git a/chrome/common/extensions/extension_file_util.cc b/chrome/common/extensions/extension_file_util.cc
|
| index 89d2e0065c4030b527320f7c7f6516cebcdadea8..efac29ecfe4ae865e894f7501ae21d6fc7770abf 100644
|
| --- a/chrome/common/extensions/extension_file_util.cc
|
| +++ b/chrome/common/extensions/extension_file_util.cc
|
| @@ -4,42 +4,19 @@
|
|
|
| #include "chrome/common/extensions/extension_file_util.h"
|
|
|
| -#include <map>
|
| -#include <vector>
|
| -
|
| -#include "base/file_util.h"
|
| -#include "base/files/file_enumerator.h"
|
| -#include "base/files/scoped_temp_dir.h"
|
| -#include "base/json/json_file_value_serializer.h"
|
| -#include "base/logging.h"
|
| -#include "base/metrics/histogram.h"
|
| -#include "base/path_service.h"
|
| -#include "base/strings/stringprintf.h"
|
| -#include "base/threading/thread_restrictions.h"
|
| -#include "chrome/common/chrome_paths.h"
|
| +#include "base/files/file_path.h"
|
| +#include "base/values.h"
|
| #include "chrome/common/extensions/api/extension_action/action_info.h"
|
| #include "chrome/common/extensions/manifest_handlers/theme_handler.h"
|
| #include "extensions/common/constants.h"
|
| #include "extensions/common/extension.h"
|
| #include "extensions/common/extension_icon_set.h"
|
| -#include "extensions/common/extension_l10n_util.h"
|
| -#include "extensions/common/extension_messages.h"
|
| -#include "extensions/common/extension_resource.h"
|
| -#include "extensions/common/install_warning.h"
|
| -#include "extensions/common/manifest.h"
|
| -#include "extensions/common/manifest_constants.h"
|
| -#include "extensions/common/manifest_handler.h"
|
| #include "extensions/common/manifest_handlers/icons_handler.h"
|
| -#include "grit/generated_resources.h"
|
| -#include "net/base/file_stream.h"
|
| -#include "ui/base/l10n/l10n_util.h"
|
|
|
| using extensions::Extension;
|
| using extensions::ExtensionResource;
|
| using extensions::Manifest;
|
|
|
| -namespace errors = extensions::manifest_errors;
|
| -
|
| namespace {
|
|
|
| // Add the image paths contained in the |icon_set| to |image_paths|.
|
| @@ -57,210 +34,6 @@ void AddPathsFromIconSet(const ExtensionIconSet& icon_set,
|
|
|
| namespace extension_file_util {
|
|
|
| -const base::FilePath::CharType kTempDirectoryName[] = FILE_PATH_LITERAL("Temp");
|
| -
|
| -base::FilePath InstallExtension(const base::FilePath& unpacked_source_dir,
|
| - const std::string& id,
|
| - const std::string& version,
|
| - const base::FilePath& extensions_dir) {
|
| - base::FilePath extension_dir = extensions_dir.AppendASCII(id);
|
| - base::FilePath version_dir;
|
| -
|
| - // Create the extension directory if it doesn't exist already.
|
| - if (!base::PathExists(extension_dir)) {
|
| - if (!base::CreateDirectory(extension_dir))
|
| - return base::FilePath();
|
| - }
|
| -
|
| - // Get a temp directory on the same file system as the profile.
|
| - base::FilePath install_temp_dir = GetInstallTempDir(extensions_dir);
|
| - base::ScopedTempDir extension_temp_dir;
|
| - if (install_temp_dir.empty() ||
|
| - !extension_temp_dir.CreateUniqueTempDirUnderPath(install_temp_dir)) {
|
| - LOG(ERROR) << "Creating of temp dir under in the profile failed.";
|
| - return base::FilePath();
|
| - }
|
| - base::FilePath crx_temp_source =
|
| - extension_temp_dir.path().Append(unpacked_source_dir.BaseName());
|
| - if (!base::Move(unpacked_source_dir, crx_temp_source)) {
|
| - LOG(ERROR) << "Moving extension from : " << unpacked_source_dir.value()
|
| - << " to : " << crx_temp_source.value() << " failed.";
|
| - return base::FilePath();
|
| - }
|
| -
|
| - // Try to find a free directory. There can be legitimate conflicts in the case
|
| - // of overinstallation of the same version.
|
| - const int kMaxAttempts = 100;
|
| - for (int i = 0; i < kMaxAttempts; ++i) {
|
| - base::FilePath candidate = extension_dir.AppendASCII(
|
| - base::StringPrintf("%s_%u", version.c_str(), i));
|
| - if (!base::PathExists(candidate)) {
|
| - version_dir = candidate;
|
| - break;
|
| - }
|
| - }
|
| -
|
| - if (version_dir.empty()) {
|
| - LOG(ERROR) << "Could not find a home for extension " << id << " with "
|
| - << "version " << version << ".";
|
| - return base::FilePath();
|
| - }
|
| -
|
| - if (!base::Move(crx_temp_source, version_dir)) {
|
| - LOG(ERROR) << "Installing extension from : " << crx_temp_source.value()
|
| - << " into : " << version_dir.value() << " failed.";
|
| - return base::FilePath();
|
| - }
|
| -
|
| - return version_dir;
|
| -}
|
| -
|
| -void UninstallExtension(const base::FilePath& extensions_dir,
|
| - const std::string& id) {
|
| - // We don't care about the return value. If this fails (and it can, due to
|
| - // plugins that aren't unloaded yet), it will get cleaned up by
|
| - // ExtensionGarbageCollector::GarbageCollectExtensions.
|
| - base::DeleteFile(extensions_dir.AppendASCII(id), true); // recursive.
|
| -}
|
| -
|
| -scoped_refptr<Extension> LoadExtension(const base::FilePath& extension_path,
|
| - Manifest::Location location,
|
| - int flags,
|
| - std::string* error) {
|
| - return LoadExtension(extension_path, std::string(), location, flags, error);
|
| -}
|
| -
|
| -scoped_refptr<Extension> LoadExtension(const base::FilePath& extension_path,
|
| - const std::string& extension_id,
|
| - Manifest::Location location,
|
| - int flags,
|
| - std::string* error) {
|
| - scoped_ptr<base::DictionaryValue> manifest(
|
| - LoadManifest(extension_path, error));
|
| - if (!manifest.get())
|
| - return NULL;
|
| - if (!extension_l10n_util::LocalizeExtension(extension_path, manifest.get(),
|
| - error)) {
|
| - return NULL;
|
| - }
|
| -
|
| - scoped_refptr<Extension> extension(Extension::Create(extension_path,
|
| - location,
|
| - *manifest,
|
| - flags,
|
| - extension_id,
|
| - error));
|
| - if (!extension.get())
|
| - return NULL;
|
| -
|
| - std::vector<extensions::InstallWarning> warnings;
|
| - if (!ValidateExtension(extension.get(), error, &warnings))
|
| - return NULL;
|
| - extension->AddInstallWarnings(warnings);
|
| -
|
| - return extension;
|
| -}
|
| -
|
| -base::DictionaryValue* LoadManifest(const base::FilePath& extension_path,
|
| - std::string* error) {
|
| - base::FilePath manifest_path =
|
| - extension_path.Append(extensions::kManifestFilename);
|
| - if (!base::PathExists(manifest_path)) {
|
| - *error = l10n_util::GetStringUTF8(IDS_EXTENSION_MANIFEST_UNREADABLE);
|
| - return NULL;
|
| - }
|
| -
|
| - JSONFileValueSerializer serializer(manifest_path);
|
| - scoped_ptr<base::Value> root(serializer.Deserialize(NULL, error));
|
| - if (!root.get()) {
|
| - if (error->empty()) {
|
| - // If |error| is empty, than the file could not be read.
|
| - // It would be cleaner to have the JSON reader give a specific error
|
| - // in this case, but other code tests for a file error with
|
| - // error->empty(). For now, be consistent.
|
| - *error = l10n_util::GetStringUTF8(IDS_EXTENSION_MANIFEST_UNREADABLE);
|
| - } else {
|
| - *error = base::StringPrintf("%s %s",
|
| - errors::kManifestParseError,
|
| - error->c_str());
|
| - }
|
| - return NULL;
|
| - }
|
| -
|
| - if (!root->IsType(base::Value::TYPE_DICTIONARY)) {
|
| - *error = l10n_util::GetStringUTF8(IDS_EXTENSION_MANIFEST_INVALID);
|
| - return NULL;
|
| - }
|
| -
|
| - return static_cast<base::DictionaryValue*>(root.release());
|
| -}
|
| -
|
| -std::vector<base::FilePath> FindPrivateKeyFiles(
|
| - const base::FilePath& extension_dir) {
|
| - std::vector<base::FilePath> result;
|
| - // Pattern matching only works at the root level, so filter manually.
|
| - base::FileEnumerator traversal(extension_dir, /*recursive=*/true,
|
| - base::FileEnumerator::FILES);
|
| - for (base::FilePath current = traversal.Next(); !current.empty();
|
| - current = traversal.Next()) {
|
| - if (!current.MatchesExtension(extensions::kExtensionKeyFileExtension))
|
| - continue;
|
| -
|
| - std::string key_contents;
|
| - if (!base::ReadFileToString(current, &key_contents)) {
|
| - // If we can't read the file, assume it's not a private key.
|
| - continue;
|
| - }
|
| - std::string key_bytes;
|
| - if (!Extension::ParsePEMKeyBytes(key_contents, &key_bytes)) {
|
| - // If we can't parse the key, assume it's ok too.
|
| - continue;
|
| - }
|
| -
|
| - result.push_back(current);
|
| - }
|
| - return result;
|
| -}
|
| -
|
| -bool ValidateExtension(const Extension* extension,
|
| - std::string* error,
|
| - std::vector<extensions::InstallWarning>* warnings) {
|
| - // Ask registered manifest handlers to validate their paths.
|
| - if (!extensions::ManifestHandler::ValidateExtension(
|
| - extension, error, warnings))
|
| - return false;
|
| -
|
| - // Check children of extension root to see if any of them start with _ and is
|
| - // not on the reserved list. We only warn, and do not block the loading of the
|
| - // extension.
|
| - std::string warning;
|
| - if (!CheckForIllegalFilenames(extension->path(), &warning))
|
| - warnings->push_back(extensions::InstallWarning(warning));
|
| -
|
| - // Check that extensions don't include private key files.
|
| - std::vector<base::FilePath> private_keys =
|
| - FindPrivateKeyFiles(extension->path());
|
| - if (extension->creation_flags() & Extension::ERROR_ON_PRIVATE_KEY) {
|
| - if (!private_keys.empty()) {
|
| - // Only print one of the private keys because l10n_util doesn't have a way
|
| - // to translate a list of strings.
|
| - *error = l10n_util::GetStringFUTF8(
|
| - IDS_EXTENSION_CONTAINS_PRIVATE_KEY,
|
| - private_keys.front().LossyDisplayName());
|
| - return false;
|
| - }
|
| - } else {
|
| - for (size_t i = 0; i < private_keys.size(); ++i) {
|
| - warnings->push_back(extensions::InstallWarning(
|
| - l10n_util::GetStringFUTF8(
|
| - IDS_EXTENSION_CONTAINS_PRIVATE_KEY,
|
| - private_keys[i].LossyDisplayName())));
|
| - }
|
| - // Only warn; don't block loading the extension.
|
| - }
|
| - return true;
|
| -}
|
| -
|
| std::set<base::FilePath> GetBrowserImagePaths(const Extension* extension) {
|
| std::set<base::FilePath> image_paths;
|
|
|
| @@ -291,77 +64,4 @@ std::set<base::FilePath> GetBrowserImagePaths(const Extension* extension) {
|
| return image_paths;
|
| }
|
|
|
| -bool CheckForIllegalFilenames(const base::FilePath& extension_path,
|
| - std::string* error) {
|
| - // Reserved underscore names.
|
| - static const base::FilePath::CharType* reserved_names[] = {
|
| - extensions::kLocaleFolder,
|
| - extensions::kPlatformSpecificFolder,
|
| - FILE_PATH_LITERAL("__MACOSX"),
|
| - };
|
| - CR_DEFINE_STATIC_LOCAL(
|
| - std::set<base::FilePath::StringType>, reserved_underscore_names,
|
| - (reserved_names, reserved_names + arraysize(reserved_names)));
|
| -
|
| - // Enumerate all files and directories in the extension root.
|
| - // There is a problem when using pattern "_*" with FileEnumerator, so we have
|
| - // to cheat with find_first_of and match all.
|
| - const int kFilesAndDirectories =
|
| - base::FileEnumerator::DIRECTORIES | base::FileEnumerator::FILES;
|
| - base::FileEnumerator all_files(extension_path, false, kFilesAndDirectories);
|
| -
|
| - base::FilePath file;
|
| - while (!(file = all_files.Next()).empty()) {
|
| - base::FilePath::StringType filename = file.BaseName().value();
|
| - // Skip all that don't start with "_".
|
| - if (filename.find_first_of(FILE_PATH_LITERAL("_")) != 0) continue;
|
| - if (reserved_underscore_names.find(filename) ==
|
| - reserved_underscore_names.end()) {
|
| - *error = base::StringPrintf(
|
| - "Cannot load extension with file or directory name %s. "
|
| - "Filenames starting with \"_\" are reserved for use by the system.",
|
| - file.BaseName().AsUTF8Unsafe().c_str());
|
| - return false;
|
| - }
|
| - }
|
| -
|
| - return true;
|
| -}
|
| -
|
| -base::FilePath GetInstallTempDir(const base::FilePath& extensions_dir) {
|
| - // We do file IO in this function, but only when the current profile's
|
| - // Temp directory has never been used before, or in a rare error case.
|
| - // Developers are not likely to see these situations often, so do an
|
| - // explicit thread check.
|
| - base::ThreadRestrictions::AssertIOAllowed();
|
| -
|
| - // Create the temp directory as a sub-directory of the Extensions directory.
|
| - // This guarantees it is on the same file system as the extension's eventual
|
| - // install target.
|
| - base::FilePath temp_path = extensions_dir.Append(kTempDirectoryName);
|
| - if (base::PathExists(temp_path)) {
|
| - if (!base::DirectoryExists(temp_path)) {
|
| - DLOG(WARNING) << "Not a directory: " << temp_path.value();
|
| - return base::FilePath();
|
| - }
|
| - if (!base::PathIsWritable(temp_path)) {
|
| - DLOG(WARNING) << "Can't write to path: " << temp_path.value();
|
| - return base::FilePath();
|
| - }
|
| - // This is a directory we can write to.
|
| - return temp_path;
|
| - }
|
| -
|
| - // Directory doesn't exist, so create it.
|
| - if (!base::CreateDirectory(temp_path)) {
|
| - DLOG(WARNING) << "Couldn't create directory: " << temp_path.value();
|
| - return base::FilePath();
|
| - }
|
| - return temp_path;
|
| -}
|
| -
|
| -void DeleteFile(const base::FilePath& path, bool recursive) {
|
| - base::DeleteFile(path, recursive);
|
| -}
|
| -
|
| } // namespace extension_file_util
|
|
|