Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 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/common/extensions/manifest.h" | 5 #include "chrome/common/extensions/manifest.h" |
| 6 | 6 |
| 7 #include "base/basictypes.h" | 7 #include "base/basictypes.h" |
| 8 #include "base/lazy_instance.h" | 8 #include "base/lazy_instance.h" |
| 9 #include "base/logging.h" | 9 #include "base/logging.h" |
| 10 #include "base/string_split.h" | 10 #include "base/string_split.h" |
| 11 #include "base/stringprintf.h" | 11 #include "base/stringprintf.h" |
| 12 #include "base/utf_string_conversions.h" | 12 #include "base/utf_string_conversions.h" |
| 13 #include "chrome/common/extensions/extension_manifest_constants.h" | 13 #include "chrome/common/extensions/extension_manifest_constants.h" |
| 14 #include "chrome/common/extensions/features/base_feature_provider.h" | 14 #include "chrome/common/extensions/features/base_feature_provider.h" |
| 15 #include "extensions/common/error_utils.h" | 15 #include "extensions/common/error_utils.h" |
| 16 | 16 |
| 17 namespace errors = extension_manifest_errors; | 17 namespace errors = extension_manifest_errors; |
| 18 namespace keys = extension_manifest_keys; | 18 namespace keys = extension_manifest_keys; |
| 19 | 19 |
| 20 namespace extensions { | 20 namespace extensions { |
| 21 | 21 |
| 22 Manifest::Manifest(Extension::Location location, | 22 namespace { |
| 23 | |
| 24 // Rank extension locations in a way that allows | |
| 25 // Manifest::GetHigherPriorityLocation() to compare locations. | |
| 26 // An extension installed from two locations will have the location | |
| 27 // with the higher rank, as returned by this function. The actual | |
| 28 // integer values may change, and should never be persisted. | |
| 29 int GetLocationRank(Manifest::Location location) { | |
| 30 const int kInvalidRank = -1; | |
| 31 int rank = kInvalidRank; // Will CHECK that rank is not kInvalidRank. | |
| 32 | |
| 33 switch (location) { | |
| 34 // Component extensions can not be overriden by any other type. | |
| 35 case Manifest::COMPONENT: | |
| 36 rank = 6; | |
| 37 break; | |
| 38 | |
| 39 // Policy controlled extensions may not be overridden by any type | |
| 40 // that is not part of chrome. | |
| 41 case Manifest::EXTERNAL_POLICY_DOWNLOAD: | |
| 42 rank = 5; | |
| 43 break; | |
| 44 | |
| 45 // A developer-loaded extension should override any installed type | |
| 46 // that a user can disable. | |
| 47 case Manifest::LOAD: | |
| 48 rank = 4; | |
| 49 break; | |
| 50 | |
| 51 // The relative priority of various external sources is not important, | |
| 52 // but having some order ensures deterministic behavior. | |
| 53 case Manifest::EXTERNAL_REGISTRY: | |
| 54 rank = 3; | |
| 55 break; | |
| 56 | |
| 57 case Manifest::EXTERNAL_PREF: | |
| 58 rank = 2; | |
| 59 break; | |
| 60 | |
| 61 case Manifest::EXTERNAL_PREF_DOWNLOAD: | |
| 62 rank = 1; | |
| 63 break; | |
| 64 | |
| 65 // User installed extensions are overridden by any external type. | |
| 66 case Manifest::INTERNAL: | |
| 67 rank = 0; | |
| 68 break; | |
| 69 | |
| 70 default: | |
| 71 NOTREACHED() << "Need to add new extension locaton " << location; | |
| 72 } | |
| 73 | |
| 74 CHECK(rank != kInvalidRank); | |
| 75 return rank; | |
| 76 } | |
| 77 | |
| 78 } // namespace | |
| 79 | |
| 80 // static | |
| 81 Manifest::Location Manifest::GetHigherPriorityLocation( | |
| 82 Manifest::Location loc1, Manifest::Location loc2) { | |
| 83 if (loc1 == loc2) | |
| 84 return loc1; | |
| 85 | |
| 86 int loc1_rank = GetLocationRank(loc1); | |
| 87 int loc2_rank = GetLocationRank(loc2); | |
| 88 | |
| 89 // If two different locations have the same rank, then we can not | |
| 90 // deterministicly choose a location. | |
| 91 CHECK(loc1_rank != loc2_rank); | |
| 92 | |
| 93 // Highest rank has highest priority. | |
| 94 return (loc1_rank > loc2_rank ? loc1 : loc2 ); | |
| 95 } | |
| 96 | |
| 97 // TODO(yoz): remove extraneous Manifest:: | |
|
Jeffrey Yasskin
2013/01/29 22:54:07
Why can't you do it now?
Yoyo Zhou
2013/01/30 01:16:01
I left this here so I would remember to actually d
| |
| 98 Manifest::Manifest(Manifest::Location location, | |
| 23 scoped_ptr<DictionaryValue> value) | 99 scoped_ptr<DictionaryValue> value) |
| 24 : location_(location), | 100 : location_(location), |
| 25 value_(value.Pass()), | 101 value_(value.Pass()), |
| 26 type_(Extension::TYPE_UNKNOWN) { | 102 type_(Manifest::TYPE_UNKNOWN) { |
| 27 if (value_->HasKey(keys::kTheme)) { | 103 if (value_->HasKey(keys::kTheme)) { |
| 28 type_ = Extension::TYPE_THEME; | 104 type_ = Manifest::TYPE_THEME; |
| 29 } else if (value_->HasKey(keys::kApp)) { | 105 } else if (value_->HasKey(keys::kApp)) { |
| 30 if (value_->Get(keys::kWebURLs, NULL) || | 106 if (value_->Get(keys::kWebURLs, NULL) || |
| 31 value_->Get(keys::kLaunchWebURL, NULL)) { | 107 value_->Get(keys::kLaunchWebURL, NULL)) { |
| 32 type_ = Extension::TYPE_HOSTED_APP; | 108 type_ = Manifest::TYPE_HOSTED_APP; |
| 33 } else if (value_->Get(keys::kPlatformAppBackground, NULL)) { | 109 } else if (value_->Get(keys::kPlatformAppBackground, NULL)) { |
| 34 type_ = Extension::TYPE_PLATFORM_APP; | 110 type_ = Manifest::TYPE_PLATFORM_APP; |
| 35 } else { | 111 } else { |
| 36 type_ = Extension::TYPE_LEGACY_PACKAGED_APP; | 112 type_ = Manifest::TYPE_LEGACY_PACKAGED_APP; |
| 37 } | 113 } |
| 38 } else { | 114 } else { |
| 39 type_ = Extension::TYPE_EXTENSION; | 115 type_ = Manifest::TYPE_EXTENSION; |
| 40 } | 116 } |
| 41 CHECK_NE(type_, Extension::TYPE_UNKNOWN); | 117 CHECK_NE(type_, Manifest::TYPE_UNKNOWN); |
| 42 } | 118 } |
| 43 | 119 |
| 44 Manifest::~Manifest() { | 120 Manifest::~Manifest() { |
| 45 } | 121 } |
| 46 | 122 |
| 47 void Manifest::ValidateManifest( | 123 void Manifest::ValidateManifest( |
| 48 std::string* error, | 124 std::string* error, |
| 49 Extension::InstallWarningVector* warnings) const { | 125 InstallWarning::Vector* warnings) const { |
| 50 *error = ""; | 126 *error = ""; |
| 51 if (type_ == Extension::TYPE_PLATFORM_APP && GetManifestVersion() < 2) { | 127 if (type_ == Manifest::TYPE_PLATFORM_APP && GetManifestVersion() < 2) { |
| 52 *error = errors::kPlatformAppNeedsManifestVersion2; | 128 *error = errors::kPlatformAppNeedsManifestVersion2; |
| 53 return; | 129 return; |
| 54 } | 130 } |
| 55 | 131 |
| 56 // Check every feature to see if its in the manifest. Note that this means | 132 // Check every feature to see if its in the manifest. Note that this means |
| 57 // we will ignore keys that are not features; we do this for forward | 133 // we will ignore keys that are not features; we do this for forward |
| 58 // compatibility. | 134 // compatibility. |
| 59 // TODO(aa): Consider having an error here in the case of strict error | 135 // TODO(aa): Consider having an error here in the case of strict error |
| 60 // checking to let developers know when they screw up. | 136 // checking to let developers know when they screw up. |
| 61 | 137 |
| 62 std::set<std::string> feature_names = | 138 std::set<std::string> feature_names = |
| 63 BaseFeatureProvider::GetManifestFeatures()->GetAllFeatureNames(); | 139 BaseFeatureProvider::GetManifestFeatures()->GetAllFeatureNames(); |
| 64 for (std::set<std::string>::iterator feature_name = feature_names.begin(); | 140 for (std::set<std::string>::iterator feature_name = feature_names.begin(); |
| 65 feature_name != feature_names.end(); ++feature_name) { | 141 feature_name != feature_names.end(); ++feature_name) { |
| 66 // Use Get instead of HasKey because the former uses path expansion. | 142 // Use Get instead of HasKey because the former uses path expansion. |
| 67 if (!value_->Get(*feature_name, NULL)) | 143 if (!value_->Get(*feature_name, NULL)) |
| 68 continue; | 144 continue; |
| 69 | 145 |
| 70 Feature* feature = | 146 Feature* feature = |
| 71 BaseFeatureProvider::GetManifestFeatures()->GetFeature(*feature_name); | 147 BaseFeatureProvider::GetManifestFeatures()->GetFeature(*feature_name); |
| 72 Feature::Availability result = feature->IsAvailableToManifest( | 148 Feature::Availability result = feature->IsAvailableToManifest( |
| 73 extension_id_, type_, Feature::ConvertLocation(location_), | 149 extension_id_, type_, Feature::ConvertLocation(location_), |
| 74 GetManifestVersion()); | 150 GetManifestVersion()); |
| 75 if (!result.is_available()) | 151 if (!result.is_available()) |
| 76 warnings->push_back(Extension::InstallWarning( | 152 warnings->push_back(InstallWarning( |
| 77 Extension::InstallWarning::FORMAT_TEXT, result.message())); | 153 InstallWarning::FORMAT_TEXT, result.message())); |
| 78 } | 154 } |
| 79 | 155 |
| 80 // Also generate warnings for keys that are not features. | 156 // Also generate warnings for keys that are not features. |
| 81 for (DictionaryValue::key_iterator key = value_->begin_keys(); | 157 for (DictionaryValue::key_iterator key = value_->begin_keys(); |
| 82 key != value_->end_keys(); ++key) { | 158 key != value_->end_keys(); ++key) { |
| 83 if (!BaseFeatureProvider::GetManifestFeatures()->GetFeature(*key)) { | 159 if (!BaseFeatureProvider::GetManifestFeatures()->GetFeature(*key)) { |
| 84 warnings->push_back(Extension::InstallWarning( | 160 warnings->push_back(InstallWarning( |
| 85 Extension::InstallWarning::FORMAT_TEXT, | 161 InstallWarning::FORMAT_TEXT, |
| 86 base::StringPrintf("Unrecognized manifest key '%s'.", | 162 base::StringPrintf("Unrecognized manifest key '%s'.", |
| 87 (*key).c_str()))); | 163 (*key).c_str()))); |
| 88 } | 164 } |
| 89 } | 165 } |
| 90 } | 166 } |
| 91 | 167 |
| 92 bool Manifest::HasKey(const std::string& key) const { | 168 bool Manifest::HasKey(const std::string& key) const { |
| 93 return CanAccessKey(key) && value_->HasKey(key); | 169 return CanAccessKey(key) && value_->HasKey(key); |
| 94 } | 170 } |
| 95 | 171 |
| (...skipping 44 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 140 return manifest; | 216 return manifest; |
| 141 } | 217 } |
| 142 | 218 |
| 143 bool Manifest::Equals(const Manifest* other) const { | 219 bool Manifest::Equals(const Manifest* other) const { |
| 144 return other && value_->Equals(other->value()); | 220 return other && value_->Equals(other->value()); |
| 145 } | 221 } |
| 146 | 222 |
| 147 int Manifest::GetManifestVersion() const { | 223 int Manifest::GetManifestVersion() const { |
| 148 // Platform apps were launched after manifest version 2 was the preferred | 224 // Platform apps were launched after manifest version 2 was the preferred |
| 149 // version, so they default to that. | 225 // version, so they default to that. |
| 150 int manifest_version = type_ == Extension::TYPE_PLATFORM_APP ? 2 : 1; | 226 int manifest_version = type_ == Manifest::TYPE_PLATFORM_APP ? 2 : 1; |
| 151 value_->GetInteger(keys::kManifestVersion, &manifest_version); | 227 value_->GetInteger(keys::kManifestVersion, &manifest_version); |
| 152 return manifest_version; | 228 return manifest_version; |
| 153 } | 229 } |
| 154 | 230 |
| 155 bool Manifest::CanAccessPath(const std::string& path) const { | 231 bool Manifest::CanAccessPath(const std::string& path) const { |
| 156 std::vector<std::string> components; | 232 std::vector<std::string> components; |
| 157 base::SplitString(path, '.', &components); | 233 base::SplitString(path, '.', &components); |
| 158 std::string key; | 234 std::string key; |
| 159 for (size_t i = 0; i < components.size(); ++i) { | 235 for (size_t i = 0; i < components.size(); ++i) { |
| 160 key += components[i]; | 236 key += components[i]; |
| 161 if (!CanAccessKey(key)) | 237 if (!CanAccessKey(key)) |
| 162 return false; | 238 return false; |
| 163 key += '.'; | 239 key += '.'; |
| 164 } | 240 } |
| 165 return true; | 241 return true; |
| 166 } | 242 } |
| 167 | 243 |
| 168 bool Manifest::CanAccessKey(const std::string& key) const { | 244 bool Manifest::CanAccessKey(const std::string& key) const { |
| 169 Feature* feature = | 245 Feature* feature = |
| 170 BaseFeatureProvider::GetManifestFeatures()->GetFeature(key); | 246 BaseFeatureProvider::GetManifestFeatures()->GetFeature(key); |
| 171 if (!feature) | 247 if (!feature) |
| 172 return true; | 248 return true; |
| 173 | 249 |
| 174 return feature->IsAvailableToManifest( | 250 return feature->IsAvailableToManifest( |
| 175 extension_id_, type_, Feature::ConvertLocation(location_), | 251 extension_id_, type_, Feature::ConvertLocation(location_), |
| 176 GetManifestVersion()).is_available(); | 252 GetManifestVersion()).is_available(); |
| 177 } | 253 } |
| 178 | 254 |
| 179 } // namespace extensions | 255 } // namespace extensions |
| OLD | NEW |