Index: chrome/common/extensions/features/base_feature_provider.cc |
diff --git a/chrome/common/extensions/features/base_feature_provider.cc b/chrome/common/extensions/features/base_feature_provider.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..d42cb50acc90375e67930af3aeeb64b086de1bd1 |
--- /dev/null |
+++ b/chrome/common/extensions/features/base_feature_provider.cc |
@@ -0,0 +1,166 @@ |
+// 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 "chrome/common/extensions/features/base_feature_provider.h" |
+ |
+#include "base/json/json_reader.h" |
+#include "base/lazy_instance.h" |
+#include "chrome/common/extensions/features/complex_feature.h" |
+#include "chrome/common/extensions/features/manifest_feature.h" |
+#include "chrome/common/extensions/features/permission_feature.h" |
+#include "grit/common_resources.h" |
+#include "ui/base/resource/resource_bundle.h" |
+ |
+namespace extensions { |
+ |
+namespace { |
+ |
+template<class FeatureClass> |
+SimpleFeature* CreateFeature() { |
+ return new FeatureClass(); |
+} |
+ |
+struct Static { |
+ Static() |
+ : manifest_features( |
+ LoadProvider("manifest", |
+ &CreateFeature<ManifestFeature>, |
+ IDR_EXTENSION_MANIFEST_FEATURES)), |
+ permission_features( |
+ LoadProvider("permissions", |
+ &CreateFeature<PermissionFeature>, |
+ IDR_EXTENSION_PERMISSION_FEATURES)) { |
+ } |
+ |
+ scoped_ptr<BaseFeatureProvider> manifest_features; |
+ scoped_ptr<BaseFeatureProvider> permission_features; |
+ |
+ private: |
+ scoped_ptr<BaseFeatureProvider> LoadProvider( |
+ const std::string& debug_string, |
+ BaseFeatureProvider::FeatureFactory factory, |
+ int resource_id) { |
+ std::string manifest_features = |
+ ResourceBundle::GetSharedInstance().GetRawDataResource( |
+ resource_id).as_string(); |
+ int error_code = 0; |
+ std::string error_message; |
+ Value* value = base::JSONReader::ReadAndReturnError( |
+ manifest_features, base::JSON_PARSE_RFC, |
+ &error_code, &error_message); |
+ CHECK(value) << "Could not load features: " << debug_string << " " |
+ << error_message; |
+ CHECK(value->IsType(Value::TYPE_DICTIONARY)) << debug_string; |
+ scoped_ptr<DictionaryValue> dictionary_value( |
+ static_cast<DictionaryValue*>(value)); |
+ return scoped_ptr<BaseFeatureProvider>( |
+ new BaseFeatureProvider(*dictionary_value, factory)); |
+ } |
+}; |
+ |
+bool ParseFeature(const DictionaryValue* value, |
+ const std::string& name, |
+ SimpleFeature* feature) { |
+ feature->set_name(name); |
+ feature->Parse(value); |
+ |
+ if (feature->extension_types()->empty()) { |
+ LOG(ERROR) << name << ": Simple features must specify at least one " |
+ << "value for extension_types."; |
+ return false; |
+ } |
+ |
+ if (!feature->GetContexts()->empty()) { |
+ LOG(ERROR) << name << ": Simple features do not support contexts."; |
+ return false; |
+ } |
+ |
+ return true; |
+} |
+ |
+base::LazyInstance<Static> g_static = LAZY_INSTANCE_INITIALIZER; |
+ |
+} // namespace |
+ |
+BaseFeatureProvider::BaseFeatureProvider(const DictionaryValue& root, |
+ FeatureFactory factory) |
+ : factory_(factory ? factory : |
+ static_cast<FeatureFactory>(&CreateFeature<SimpleFeature>)) { |
+ for (DictionaryValue::Iterator iter(root); iter.HasNext(); iter.Advance()) { |
+ if (iter.value().GetType() == Value::TYPE_DICTIONARY) { |
+ linked_ptr<SimpleFeature> feature((*factory_)()); |
+ |
+ if (!ParseFeature(static_cast<const DictionaryValue*>(&iter.value()), |
+ iter.key(), |
+ feature.get())) |
+ continue; |
+ |
+ features_[iter.key()] = feature; |
+ } else if (iter.value().GetType() == Value::TYPE_LIST) { |
+ // This is a complex feature. |
+ const ListValue* list = static_cast<const ListValue*>(&iter.value()); |
+ CHECK_GT(list->GetSize(), 0UL); |
+ |
+ scoped_ptr<ComplexFeature::FeatureList> features( |
+ new ComplexFeature::FeatureList()); |
+ |
+ // Parse and add all SimpleFeatures from the list. |
+ for (ListValue::const_iterator list_iter = list->begin(); |
+ list_iter != list->end(); ++list_iter) { |
+ if ((*list_iter)->GetType() != Value::TYPE_DICTIONARY) { |
+ LOG(ERROR) << iter.key() << ": Feature rules must be dictionaries."; |
+ continue; |
+ } |
+ |
+ scoped_ptr<SimpleFeature> feature((*factory_)()); |
+ if (!ParseFeature(static_cast<const DictionaryValue*>(*list_iter), |
+ iter.key(), |
+ feature.get())) |
+ continue; |
+ |
+ features->push_back(feature.release()); |
+ } |
+ |
+ linked_ptr<ComplexFeature> feature(new ComplexFeature(features.Pass())); |
+ feature->set_name(iter.key()); |
+ |
+ features_[iter.key()] = feature; |
+ } else { |
+ LOG(ERROR) << iter.key() << ": Feature description must be dictionary or" |
+ << " list of dictionaries."; |
+ } |
+ } |
+} |
+ |
+BaseFeatureProvider::~BaseFeatureProvider() { |
+} |
+ |
+// static |
+BaseFeatureProvider* BaseFeatureProvider::GetManifestFeatures() { |
+ return g_static.Get().manifest_features.get(); |
+} |
+ |
+// static |
+BaseFeatureProvider* BaseFeatureProvider::GetPermissionFeatures() { |
+ return g_static.Get().permission_features.get(); |
+} |
+ |
+std::set<std::string> BaseFeatureProvider::GetAllFeatureNames() const { |
+ std::set<std::string> result; |
+ for (FeatureMap::const_iterator iter = features_.begin(); |
+ iter != features_.end(); ++iter) { |
+ result.insert(iter->first); |
+ } |
+ return result; |
+} |
+ |
+Feature* BaseFeatureProvider::GetFeature(const std::string& name) { |
+ FeatureMap::iterator iter = features_.find(name); |
+ if (iter != features_.end()) |
+ return iter->second.get(); |
+ else |
+ return NULL; |
+} |
+ |
+} // namespace |