// Copyright 2013 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 "extensions/common/extension_api.h" #include #include #include #include #include #include "base/json/json_reader.h" #include "base/json/json_writer.h" #include "base/lazy_instance.h" #include "base/logging.h" #include "base/macros.h" #include "base/strings/string_number_conversions.h" #include "base/strings/string_split.h" #include "base/strings/string_util.h" #include "base/values.h" #include "extensions/common/extension.h" #include "extensions/common/extensions_client.h" #include "extensions/common/features/feature.h" #include "extensions/common/features/feature_provider.h" #include "extensions/common/features/simple_feature.h" #include "extensions/common/permissions/permission_set.h" #include "extensions/common/permissions/permissions_data.h" #include "extensions/grit/extensions_resources.h" #include "ui/base/resource/resource_bundle.h" #include "url/gurl.h" namespace extensions { namespace { const char* kChildKinds[] = { "functions", "events" }; base::StringPiece ReadFromResource(int resource_id) { return ResourceBundle::GetSharedInstance().GetRawDataResource( resource_id); } scoped_ptr LoadSchemaList(const std::string& name, const base::StringPiece& schema) { std::string error_message; scoped_ptr result( base::JSONReader::ReadAndReturnError( schema, base::JSON_PARSE_RFC | base::JSON_DETACHABLE_CHILDREN, // options NULL, // error code &error_message)); // Tracking down http://crbug.com/121424 char buf[128]; base::snprintf(buf, arraysize(buf), "%s: (%d) '%s'", name.c_str(), result.get() ? result->GetType() : -1, error_message.c_str()); CHECK(result.get()) << error_message << " for schema " << schema; CHECK(result->IsType(base::Value::TYPE_LIST)) << " for schema " << schema; return base::ListValue::From(std::move(result)); } const base::DictionaryValue* FindListItem(const base::ListValue* list, const std::string& property_name, const std::string& property_value) { for (size_t i = 0; i < list->GetSize(); ++i) { const base::DictionaryValue* item = NULL; CHECK(list->GetDictionary(i, &item)) << property_value << "/" << property_name; std::string value; if (item->GetString(property_name, &value) && value == property_value) return item; } return NULL; } const base::DictionaryValue* GetSchemaChild( const base::DictionaryValue* schema_node, const std::string& child_name) { const base::DictionaryValue* child_node = NULL; for (size_t i = 0; i < arraysize(kChildKinds); ++i) { const base::ListValue* list_node = NULL; if (!schema_node->GetList(kChildKinds[i], &list_node)) continue; child_node = FindListItem(list_node, "name", child_name); if (child_node) return child_node; } return NULL; } struct Static { Static() : api(ExtensionAPI::CreateWithDefaultConfiguration()) { } scoped_ptr api; }; base::LazyInstance g_lazy_instance = LAZY_INSTANCE_INITIALIZER; // May override |g_lazy_instance| for a test. ExtensionAPI* g_shared_instance_for_test = NULL; // If it exists and does not already specify a namespace, then the value stored // with key |key| in |schema| will be updated to |schema_namespace| + "." + // |schema[key]|. void MaybePrefixFieldWithNamespace(const std::string& schema_namespace, base::DictionaryValue* schema, const std::string& key) { if (!schema->HasKey(key)) return; std::string old_id; CHECK(schema->GetString(key, &old_id)); if (old_id.find(".") == std::string::npos) schema->SetString(key, schema_namespace + "." + old_id); } // Modify all "$ref" keys anywhere in |schema| to be prefxied by // |schema_namespace| if they do not already specify a namespace. void PrefixRefsWithNamespace(const std::string& schema_namespace, base::Value* value) { base::ListValue* list = NULL; base::DictionaryValue* dict = NULL; if (value->GetAsList(&list)) { for (base::ListValue::iterator i = list->begin(); i != list->end(); ++i) { PrefixRefsWithNamespace(schema_namespace, *i); } } else if (value->GetAsDictionary(&dict)) { MaybePrefixFieldWithNamespace(schema_namespace, dict, "$ref"); for (base::DictionaryValue::Iterator i(*dict); !i.IsAtEnd(); i.Advance()) { base::Value* value = NULL; CHECK(dict->GetWithoutPathExpansion(i.key(), &value)); PrefixRefsWithNamespace(schema_namespace, value); } } } // Modify all objects in the "types" section of the schema to be prefixed by // |schema_namespace| if they do not already specify a namespace. void PrefixTypesWithNamespace(const std::string& schema_namespace, base::DictionaryValue* schema) { if (!schema->HasKey("types")) return; // Add the namespace to all of the types defined in this schema base::ListValue *types = NULL; CHECK(schema->GetList("types", &types)); for (size_t i = 0; i < types->GetSize(); ++i) { base::DictionaryValue *type = NULL; CHECK(types->GetDictionary(i, &type)); MaybePrefixFieldWithNamespace(schema_namespace, type, "id"); MaybePrefixFieldWithNamespace(schema_namespace, type, "customBindings"); } } // Modify the schema so that all types are fully qualified. void PrefixWithNamespace(const std::string& schema_namespace, base::DictionaryValue* schema) { PrefixTypesWithNamespace(schema_namespace, schema); PrefixRefsWithNamespace(schema_namespace, schema); } } // namespace // static ExtensionAPI* ExtensionAPI::GetSharedInstance() { return g_shared_instance_for_test ? g_shared_instance_for_test : g_lazy_instance.Get().api.get(); } // static ExtensionAPI* ExtensionAPI::CreateWithDefaultConfiguration() { ExtensionAPI* api = new ExtensionAPI(); api->InitDefaultConfiguration(); return api; } // static void ExtensionAPI::SplitDependencyName(const std::string& full_name, std::string* feature_type, std::string* feature_name) { size_t colon_index = full_name.find(':'); if (colon_index == std::string::npos) { // TODO(aa): Remove this code when all API descriptions have been updated. *feature_type = "api"; *feature_name = full_name; return; } *feature_type = full_name.substr(0, colon_index); *feature_name = full_name.substr(colon_index + 1); } ExtensionAPI::OverrideSharedInstanceForTest::OverrideSharedInstanceForTest( ExtensionAPI* testing_api) : original_api_(g_shared_instance_for_test) { g_shared_instance_for_test = testing_api; } ExtensionAPI::OverrideSharedInstanceForTest::~OverrideSharedInstanceForTest() { g_shared_instance_for_test = original_api_; } void ExtensionAPI::LoadSchema(const std::string& name, const base::StringPiece& schema) { scoped_ptr schema_list(LoadSchemaList(name, schema)); std::string schema_namespace; extensions::ExtensionsClient* extensions_client = extensions::ExtensionsClient::Get(); DCHECK(extensions_client); while (!schema_list->empty()) { base::DictionaryValue* schema = NULL; { scoped_ptr value; schema_list->Remove(schema_list->GetSize() - 1, &value); CHECK(value.release()->GetAsDictionary(&schema)); } CHECK(schema->GetString("namespace", &schema_namespace)); PrefixWithNamespace(schema_namespace, schema); schemas_[schema_namespace] = make_linked_ptr(schema); if (!extensions_client->IsAPISchemaGenerated(schema_namespace)) CHECK_EQ(1u, unloaded_schemas_.erase(schema_namespace)); } } ExtensionAPI::ExtensionAPI() : default_configuration_initialized_(false) { } ExtensionAPI::~ExtensionAPI() { } void ExtensionAPI::InitDefaultConfiguration() { const char* names[] = {"api", "manifest", "permission"}; for (size_t i = 0; i < arraysize(names); ++i) RegisterDependencyProvider(names[i], FeatureProvider::GetByName(names[i])); ExtensionsClient::Get()->RegisterAPISchemaResources(this); RegisterSchemaResource("declarativeWebRequest", IDR_EXTENSION_API_JSON_DECLARATIVE_WEBREQUEST); RegisterSchemaResource("webViewRequest", IDR_EXTENSION_API_JSON_WEB_VIEW_REQUEST); default_configuration_initialized_ = true; } void ExtensionAPI::RegisterSchemaResource(const std::string& name, int resource_id) { unloaded_schemas_[name] = resource_id; } void ExtensionAPI::RegisterDependencyProvider(const std::string& name, const FeatureProvider* provider) { dependency_providers_[name] = provider; } bool ExtensionAPI::IsAnyFeatureAvailableToContext(const Feature& api, const Extension* extension, Feature::Context context, const GURL& url) { FeatureProviderMap::iterator provider = dependency_providers_.find("api"); CHECK(provider != dependency_providers_.end()); if (api.IsAvailableToContext(extension, context, url).is_available()) return true; // Check to see if there are any parts of this API that are allowed in this // context. const std::vector features = provider->second->GetChildren(api); for (std::vector::const_iterator it = features.begin(); it != features.end(); ++it) { if ((*it)->IsAvailableToContext(extension, context, url).is_available()) return true; } return false; } Feature::Availability ExtensionAPI::IsAvailable(const std::string& full_name, const Extension* extension, Feature::Context context, const GURL& url) { Feature* feature = GetFeatureDependency(full_name); if (!feature) { return Feature::Availability(Feature::NOT_PRESENT, std::string("Unknown feature: ") + full_name); } return feature->IsAvailableToContext(extension, context, url); } bool ExtensionAPI::IsAvailableToWebUI(const std::string& name, const GURL& url) { return IsAvailable(name, NULL, Feature::WEBUI_CONTEXT, url).is_available(); } const base::DictionaryValue* ExtensionAPI::GetSchema( const std::string& full_name) { std::string child_name; std::string api_name = GetAPINameFromFullName(full_name, &child_name); const base::DictionaryValue* result = NULL; SchemaMap::iterator maybe_schema = schemas_.find(api_name); if (maybe_schema != schemas_.end()) { result = maybe_schema->second.get(); } else { // Might not have loaded yet; or might just not exist. UnloadedSchemaMap::iterator maybe_schema_resource = unloaded_schemas_.find(api_name); extensions::ExtensionsClient* extensions_client = extensions::ExtensionsClient::Get(); DCHECK(extensions_client); if (maybe_schema_resource != unloaded_schemas_.end()) { LoadSchema(maybe_schema_resource->first, ReadFromResource(maybe_schema_resource->second)); } else if (default_configuration_initialized_ && extensions_client->IsAPISchemaGenerated(api_name)) { LoadSchema(api_name, extensions_client->GetAPISchema(api_name)); } else { return NULL; } maybe_schema = schemas_.find(api_name); CHECK(schemas_.end() != maybe_schema); result = maybe_schema->second.get(); } if (!child_name.empty()) result = GetSchemaChild(result, child_name); return result; } Feature* ExtensionAPI::GetFeatureDependency(const std::string& full_name) { std::string feature_type; std::string feature_name; SplitDependencyName(full_name, &feature_type, &feature_name); FeatureProviderMap::iterator provider = dependency_providers_.find(feature_type); if (provider == dependency_providers_.end()) return NULL; Feature* feature = provider->second->GetFeature(feature_name); // Try getting the feature for the parent API, if this was a child. if (!feature) { std::string child_name; feature = provider->second->GetFeature( GetAPINameFromFullName(feature_name, &child_name)); } return feature; } std::string ExtensionAPI::GetAPINameFromFullName(const std::string& full_name, std::string* child_name) { std::string api_name_candidate = full_name; extensions::ExtensionsClient* extensions_client = extensions::ExtensionsClient::Get(); DCHECK(extensions_client); while (true) { if (schemas_.find(api_name_candidate) != schemas_.end() || extensions_client->IsAPISchemaGenerated(api_name_candidate) || unloaded_schemas_.find(api_name_candidate) != unloaded_schemas_.end()) { std::string result = api_name_candidate; if (child_name) { if (result.length() < full_name.length()) *child_name = full_name.substr(result.length() + 1); else *child_name = ""; } return result; } size_t last_dot_index = api_name_candidate.rfind('.'); if (last_dot_index == std::string::npos) break; api_name_candidate = api_name_candidate.substr(0, last_dot_index); } *child_name = ""; return std::string(); } } // namespace extensions