| OLD | NEW |
| 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "chrome/browser/extensions/component_loader.h" | 5 #include "chrome/browser/extensions/component_loader.h" |
| 6 | 6 |
| 7 #include "base/command_line.h" | 7 #include "base/command_line.h" |
| 8 #include "base/file_util.h" |
| 8 #include "base/path_service.h" | 9 #include "base/path_service.h" |
| 9 #include "base/json/json_value_serializer.h" | 10 #include "base/json/json_value_serializer.h" |
| 10 #include "chrome/browser/browser_process.h" | 11 #include "chrome/browser/browser_process.h" |
| 11 #include "chrome/browser/extensions/extension_service.h" | 12 #include "chrome/browser/extensions/extension_service.h" |
| 12 #include "chrome/browser/prefs/pref_change_registrar.h" | 13 #include "chrome/browser/prefs/pref_change_registrar.h" |
| 13 #include "chrome/browser/prefs/pref_notifier.h" | 14 #include "chrome/browser/prefs/pref_notifier.h" |
| 14 #include "chrome/browser/profiles/profile.h" | 15 #include "chrome/browser/profiles/profile.h" |
| 15 #include "chrome/common/chrome_notification_types.h" | 16 #include "chrome/common/chrome_notification_types.h" |
| 16 #include "chrome/common/chrome_paths.h" | 17 #include "chrome/common/chrome_paths.h" |
| 17 #include "chrome/common/chrome_switches.h" | 18 #include "chrome/common/chrome_switches.h" |
| 18 #include "chrome/common/extensions/extension.h" | 19 #include "chrome/common/extensions/extension.h" |
| 20 #include "chrome/common/extensions/extension_file_util.h" |
| 19 #include "chrome/common/pref_names.h" | 21 #include "chrome/common/pref_names.h" |
| 20 #include "content/public/browser/notification_details.h" | 22 #include "content/public/browser/notification_details.h" |
| 21 #include "content/public/browser/notification_source.h" | 23 #include "content/public/browser/notification_source.h" |
| 22 #include "grit/browser_resources.h" | 24 #include "grit/browser_resources.h" |
| 23 #include "ui/base/resource/resource_bundle.h" | 25 #include "ui/base/resource/resource_bundle.h" |
| 24 | 26 |
| 25 #if defined(OFFICIAL_BUILD) | 27 #if defined(OFFICIAL_BUILD) |
| 26 #include "chrome/browser/defaults.h" | 28 #include "chrome/browser/defaults.h" |
| 27 #endif | 29 #endif |
| 28 | 30 |
| (...skipping 63 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 92 // required in case LoadAll() is called again. | 94 // required in case LoadAll() is called again. |
| 93 DictionaryValue* manifest = ParseManifest(manifest_contents); | 95 DictionaryValue* manifest = ParseManifest(manifest_contents); |
| 94 if (manifest) | 96 if (manifest) |
| 95 return Add(manifest, root_directory); | 97 return Add(manifest, root_directory); |
| 96 return NULL; | 98 return NULL; |
| 97 } | 99 } |
| 98 | 100 |
| 99 const Extension* ComponentLoader::Add( | 101 const Extension* ComponentLoader::Add( |
| 100 const DictionaryValue* parsed_manifest, | 102 const DictionaryValue* parsed_manifest, |
| 101 const FilePath& root_directory) { | 103 const FilePath& root_directory) { |
| 102 // Get the absolute path to the extension. | 104 CHECK(!Exists(GenerateId(parsed_manifest))); |
| 103 FilePath absolute_path(root_directory); | 105 ComponentExtensionInfo info(parsed_manifest, root_directory); |
| 104 if (!absolute_path.IsAbsolute()) { | |
| 105 if (PathService::Get(chrome::DIR_RESOURCES, &absolute_path)) { | |
| 106 absolute_path = absolute_path.Append(root_directory); | |
| 107 } else { | |
| 108 NOTREACHED(); | |
| 109 } | |
| 110 } | |
| 111 | |
| 112 ComponentExtensionInfo info(parsed_manifest, absolute_path); | |
| 113 component_extensions_.push_back(info); | 106 component_extensions_.push_back(info); |
| 114 if (extension_service_->is_ready()) | 107 if (extension_service_->is_ready()) |
| 115 return Load(info); | 108 return Load(info); |
| 116 return NULL; | 109 return NULL; |
| 117 } | 110 } |
| 118 | 111 |
| 112 const Extension* ComponentLoader::AddOrReplace(const FilePath& path) { |
| 113 FilePath absolute_path = path; |
| 114 file_util::AbsolutePath(&absolute_path); |
| 115 std::string error; |
| 116 scoped_ptr<DictionaryValue> manifest( |
| 117 extension_file_util::LoadManifest(absolute_path, &error)); |
| 118 if (!manifest.get()) { |
| 119 LOG(ERROR) << "Could not load extension from '" << |
| 120 absolute_path.value() << "'. " << error; |
| 121 return NULL; |
| 122 } |
| 123 Remove(GenerateId(manifest.get())); |
| 124 |
| 125 return Add(manifest.release(), absolute_path); |
| 126 } |
| 127 |
| 119 const Extension* ComponentLoader::Load(const ComponentExtensionInfo& info) { | 128 const Extension* ComponentLoader::Load(const ComponentExtensionInfo& info) { |
| 120 int flags = Extension::REQUIRE_KEY; | 129 int flags = Extension::REQUIRE_KEY; |
| 121 // TODO(abarth): We should REQUIRE_MODERN_MANIFEST_VERSION once we've updated | 130 // TODO(abarth): We should REQUIRE_MODERN_MANIFEST_VERSION once we've updated |
| 122 // our component extensions to the new manifest version. | 131 // our component extensions to the new manifest version. |
| 123 if (Extension::ShouldDoStrictErrorChecking(Extension::COMPONENT)) | 132 if (Extension::ShouldDoStrictErrorChecking(Extension::COMPONENT)) |
| 124 flags |= Extension::STRICT_ERROR_CHECKS; | 133 flags |= Extension::STRICT_ERROR_CHECKS; |
| 125 std::string error; | 134 std::string error; |
| 135 |
| 136 // Get the absolute path to the extension. |
| 137 FilePath absolute_path(info.root_directory); |
| 138 if (!absolute_path.IsAbsolute()) { |
| 139 if (PathService::Get(chrome::DIR_RESOURCES, &absolute_path)) { |
| 140 absolute_path = absolute_path.Append(info.root_directory); |
| 141 } else { |
| 142 NOTREACHED(); |
| 143 } |
| 144 } |
| 145 |
| 126 scoped_refptr<const Extension> extension(Extension::Create( | 146 scoped_refptr<const Extension> extension(Extension::Create( |
| 127 info.root_directory, | 147 absolute_path, |
| 128 Extension::COMPONENT, | 148 Extension::COMPONENT, |
| 129 *info.manifest, | 149 *info.manifest, |
| 130 flags, | 150 flags, |
| 131 &error)); | 151 &error)); |
| 132 if (!extension.get()) { | 152 if (!extension.get()) { |
| 133 LOG(ERROR) << error; | 153 LOG(ERROR) << error; |
| 134 return NULL; | 154 return NULL; |
| 135 } | 155 } |
| 136 extension_service_->AddExtension(extension); | 156 extension_service_->AddExtension(extension); |
| 137 return extension; | 157 return extension; |
| 138 } | 158 } |
| 139 | 159 |
| 140 void ComponentLoader::Remove(const FilePath& root_directory) { | 160 void ComponentLoader::Remove(const FilePath& root_directory) { |
| 141 // Find the ComponentExtensionInfo for the extension. | 161 // Find the ComponentExtensionInfo for the extension. |
| 142 RegisteredComponentExtensions::iterator it = component_extensions_.begin(); | 162 RegisteredComponentExtensions::iterator it = component_extensions_.begin(); |
| 143 for (; it != component_extensions_.end(); ++it) { | 163 for (; it != component_extensions_.end(); ++it) { |
| 144 if (it->root_directory == root_directory) | 164 if (it->root_directory == root_directory) { |
| 165 Remove(GenerateId(it->manifest)); |
| 145 break; | 166 break; |
| 167 } |
| 146 } | 168 } |
| 147 // If the extension is not in the list, there's nothing to do. | 169 } |
| 148 if (it == component_extensions_.end()) | |
| 149 return; | |
| 150 | 170 |
| 151 // The list owns the dictionary, so it must be deleted after removal. | 171 void ComponentLoader::Remove(const std::string& id) { |
| 152 scoped_ptr<const DictionaryValue> manifest(it->manifest); | 172 RegisteredComponentExtensions::iterator it = component_extensions_.begin(); |
| 173 for (; it != component_extensions_.end(); ++it) { |
| 174 if (GenerateId(it->manifest) == id) { |
| 175 delete it->manifest; |
| 176 it = component_extensions_.erase(it); |
| 177 if (extension_service_->is_ready()) |
| 178 extension_service_-> |
| 179 UnloadExtension(id, extension_misc::UNLOAD_REASON_DISABLE); |
| 180 break; |
| 181 } |
| 182 } |
| 183 } |
| 153 | 184 |
| 154 // Remove the extension from the list of registered extensions. | 185 bool ComponentLoader::Exists(const std::string& id) const { |
| 155 *it = component_extensions_.back(); | 186 RegisteredComponentExtensions::const_iterator it = |
| 156 component_extensions_.pop_back(); | 187 component_extensions_.begin(); |
| 188 for (; it != component_extensions_.end(); ++it) |
| 189 if (GenerateId(it->manifest) == id) |
| 190 return true; |
| 191 return false; |
| 192 } |
| 157 | 193 |
| 158 // Determine the extension id and unload the extension. | 194 std::string ComponentLoader::GenerateId(const DictionaryValue* manifest) { |
| 159 std::string public_key; | 195 std::string public_key; |
| 160 std::string public_key_bytes; | 196 std::string public_key_bytes; |
| 161 std::string id; | 197 std::string id; |
| 162 if (!manifest->GetString( | 198 if (!manifest->GetString( |
| 163 extension_manifest_keys::kPublicKey, &public_key) || | 199 extension_manifest_keys::kPublicKey, &public_key) || |
| 164 !Extension::ParsePEMKeyBytes(public_key, &public_key_bytes) || | 200 !Extension::ParsePEMKeyBytes(public_key, &public_key_bytes) || |
| 165 !Extension::GenerateId(public_key_bytes, &id)) { | 201 !Extension::GenerateId(public_key_bytes, &id)) { |
| 166 LOG(ERROR) << "Failed to get extension id"; | 202 return std::string(); |
| 167 return; | |
| 168 } | 203 } |
| 169 extension_service_-> | 204 return id; |
| 170 UnloadExtension(id, extension_misc::UNLOAD_REASON_DISABLE); | |
| 171 } | 205 } |
| 172 | 206 |
| 173 void ComponentLoader::AddFileManagerExtension() { | 207 void ComponentLoader::AddFileManagerExtension() { |
| 174 #if defined(FILE_MANAGER_EXTENSION) | 208 #if defined(FILE_MANAGER_EXTENSION) |
| 175 #ifndef NDEBUG | 209 #ifndef NDEBUG |
| 176 const CommandLine* command_line = CommandLine::ForCurrentProcess(); | 210 const CommandLine* command_line = CommandLine::ForCurrentProcess(); |
| 177 if (command_line->HasSwitch(switches::kFileManagerExtensionPath)) { | 211 if (command_line->HasSwitch(switches::kFileManagerExtensionPath)) { |
| 178 FilePath filemgr_extension_path( | 212 FilePath filemgr_extension_path( |
| 179 command_line->GetSwitchValuePath(switches::kFileManagerExtensionPath)); | 213 command_line->GetSwitchValuePath(switches::kFileManagerExtensionPath)); |
| 180 Add(IDR_FILEMANAGER_MANIFEST, filemgr_extension_path); | 214 Add(IDR_FILEMANAGER_MANIFEST, filemgr_extension_path); |
| (...skipping 106 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 287 void ComponentLoader::RegisterUserPrefs(PrefService* prefs) { | 321 void ComponentLoader::RegisterUserPrefs(PrefService* prefs) { |
| 288 prefs->RegisterStringPref(prefs::kEnterpriseWebStoreURL, | 322 prefs->RegisterStringPref(prefs::kEnterpriseWebStoreURL, |
| 289 std::string() /* default_value */, | 323 std::string() /* default_value */, |
| 290 PrefService::UNSYNCABLE_PREF); | 324 PrefService::UNSYNCABLE_PREF); |
| 291 prefs->RegisterStringPref(prefs::kEnterpriseWebStoreName, | 325 prefs->RegisterStringPref(prefs::kEnterpriseWebStoreName, |
| 292 std::string() /* default_value */, | 326 std::string() /* default_value */, |
| 293 PrefService::UNSYNCABLE_PREF); | 327 PrefService::UNSYNCABLE_PREF); |
| 294 } | 328 } |
| 295 | 329 |
| 296 } // namespace extensions | 330 } // namespace extensions |
| OLD | NEW |