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 3 // found in the LICENSE file. 4 5 #include "chrome/common/extensions/api/extension_api.h" 6 7 #include <algorithm> 8 #include <string> 9 #include <vector> 10 11 #include "base/json/json_reader.h" 12 #include "base/json/json_writer.h" 13 #include "base/lazy_instance.h" 14 #include "base/logging.h" 15 #include "base/strings/string_number_conversions.h" 16 #include "base/strings/string_split.h" 17 #include "base/strings/string_util.h" 18 #include "base/values.h" 19 #include "chrome/common/extensions/api/generated_schemas.h" 20 #include "chrome/common/extensions/extension.h" 21 #include "chrome/common/extensions/features/feature.h" 22 #include "chrome/common/extensions/permissions/permission_set.h" 23 #include "chrome/common/extensions/permissions/permissions_data.h" 24 #include "extensions/common/features/feature_provider.h" 25 #include "grit/common_resources.h" 26 #include "grit/extensions_api_resources.h" 27 #include "ui/base/resource/resource_bundle.h" 28 #include "url/gurl.h" 29 30 namespace extensions { 31 32 using api::GeneratedSchemas; 33 34 namespace { 35 36 const char* kChildKinds[] = { 37 "functions", 38 "events" 39 }; 40 41 base::StringPiece ReadFromResource(int resource_id) { 42 return ResourceBundle::GetSharedInstance().GetRawDataResource( 43 resource_id); 44 } 45 46 scoped_ptr<base::ListValue> LoadSchemaList(const std::string& name, 47 const base::StringPiece& schema) { 48 std::string error_message; 49 scoped_ptr<base::Value> result( 50 base::JSONReader::ReadAndReturnError( 51 schema, 52 base::JSON_PARSE_RFC | base::JSON_DETACHABLE_CHILDREN, // options 53 NULL, // error code 54 &error_message)); 55 56 // Tracking down http://crbug.com/121424 57 char buf[128]; 58 base::snprintf(buf, arraysize(buf), "%s: (%d) '%s'", 59 name.c_str(), 60 result.get() ? result->GetType() : -1, 61 error_message.c_str()); 62 63 CHECK(result.get()) << error_message << " for schema " << schema; 64 CHECK(result->IsType(base::Value::TYPE_LIST)) << " for schema " << schema; 65 return scoped_ptr<base::ListValue>(static_cast<base::ListValue*>( 66 result.release())); 67 } 68 69 const base::DictionaryValue* FindListItem(const base::ListValue* list, 70 const std::string& property_name, 71 const std::string& property_value) { 72 for (size_t i = 0; i < list->GetSize(); ++i) { 73 const base::DictionaryValue* item = NULL; 74 CHECK(list->GetDictionary(i, &item)) 75 << property_value << "/" << property_name; 76 std::string value; 77 if (item->GetString(property_name, &value) && value == property_value) 78 return item; 79 } 80 81 return NULL; 82 } 83 84 const base::DictionaryValue* GetSchemaChild( 85 const base::DictionaryValue* schema_node, 86 const std::string& child_name) { 87 const base::DictionaryValue* child_node = NULL; 88 for (size_t i = 0; i < arraysize(kChildKinds); ++i) { 89 const base::ListValue* list_node = NULL; 90 if (!schema_node->GetList(kChildKinds[i], &list_node)) 91 continue; 92 child_node = FindListItem(list_node, "name", child_name); 93 if (child_node) 94 return child_node; 95 } 96 97 return NULL; 98 } 99 100 struct Static { 101 Static() 102 : api(ExtensionAPI::CreateWithDefaultConfiguration()) { 103 } 104 scoped_ptr<ExtensionAPI> api; 105 }; 106 107 base::LazyInstance<Static> g_lazy_instance = LAZY_INSTANCE_INITIALIZER; 108 109 // If it exists and does not already specify a namespace, then the value stored 110 // with key |key| in |schema| will be updated to |schema_namespace| + "." + 111 // |schema[key]|. 112 void MaybePrefixFieldWithNamespace(const std::string& schema_namespace, 113 base::DictionaryValue* schema, 114 const std::string& key) { 115 if (!schema->HasKey(key)) 116 return; 117 118 std::string old_id; 119 CHECK(schema->GetString(key, &old_id)); 120 if (old_id.find(".") == std::string::npos) 121 schema->SetString(key, schema_namespace + "." + old_id); 122 } 123 124 // Modify all "$ref" keys anywhere in |schema| to be prefxied by 125 // |schema_namespace| if they do not already specify a namespace. 126 void PrefixRefsWithNamespace(const std::string& schema_namespace, 127 base::Value* value) { 128 base::ListValue* list = NULL; 129 base::DictionaryValue* dict = NULL; 130 if (value->GetAsList(&list)) { 131 for (base::ListValue::iterator i = list->begin(); i != list->end(); ++i) { 132 PrefixRefsWithNamespace(schema_namespace, *i); 133 } 134 } else if (value->GetAsDictionary(&dict)) { 135 MaybePrefixFieldWithNamespace(schema_namespace, dict, "$ref"); 136 for (base::DictionaryValue::Iterator i(*dict); !i.IsAtEnd(); i.Advance()) { 137 base::Value* value = NULL; 138 CHECK(dict->GetWithoutPathExpansion(i.key(), &value)); 139 PrefixRefsWithNamespace(schema_namespace, value); 140 } 141 } 142 } 143 144 // Modify all objects in the "types" section of the schema to be prefixed by 145 // |schema_namespace| if they do not already specify a namespace. 146 void PrefixTypesWithNamespace(const std::string& schema_namespace, 147 base::DictionaryValue* schema) { 148 if (!schema->HasKey("types")) 149 return; 150 151 // Add the namespace to all of the types defined in this schema 152 base::ListValue *types = NULL; 153 CHECK(schema->GetList("types", &types)); 154 for (size_t i = 0; i < types->GetSize(); ++i) { 155 base::DictionaryValue *type = NULL; 156 CHECK(types->GetDictionary(i, &type)); 157 MaybePrefixFieldWithNamespace(schema_namespace, type, "id"); 158 MaybePrefixFieldWithNamespace(schema_namespace, type, "customBindings"); 159 } 160 } 161 162 // Modify the schema so that all types are fully qualified. 163 void PrefixWithNamespace(const std::string& schema_namespace, 164 base::DictionaryValue* schema) { 165 PrefixTypesWithNamespace(schema_namespace, schema); 166 PrefixRefsWithNamespace(schema_namespace, schema); 167 } 168 169 } // namespace 170 171 // static 172 ExtensionAPI* ExtensionAPI::GetSharedInstance() { 173 return g_lazy_instance.Get().api.get(); 174 } 175 176 // static 177 ExtensionAPI* ExtensionAPI::CreateWithDefaultConfiguration() { 178 ExtensionAPI* api = new ExtensionAPI(); 179 api->InitDefaultConfiguration(); 180 return api; 181 } 182 183 // static 184 void ExtensionAPI::SplitDependencyName(const std::string& full_name, 185 std::string* feature_type, 186 std::string* feature_name) { 187 size_t colon_index = full_name.find(':'); 188 if (colon_index == std::string::npos) { 189 // TODO(aa): Remove this code when all API descriptions have been updated. 190 *feature_type = "api"; 191 *feature_name = full_name; 192 return; 193 } 194 195 *feature_type = full_name.substr(0, colon_index); 196 *feature_name = full_name.substr(colon_index + 1); 197 } 198 199 void ExtensionAPI::LoadSchema(const std::string& name, 200 const base::StringPiece& schema) { 201 scoped_ptr<base::ListValue> schema_list(LoadSchemaList(name, schema)); 202 std::string schema_namespace; 203 204 while (!schema_list->empty()) { 205 base::DictionaryValue* schema = NULL; 206 { 207 scoped_ptr<base::Value> value; 208 schema_list->Remove(schema_list->GetSize() - 1, &value); 209 CHECK(value.release()->GetAsDictionary(&schema)); 210 } 211 212 CHECK(schema->GetString("namespace", &schema_namespace)); 213 PrefixWithNamespace(schema_namespace, schema); 214 schemas_[schema_namespace] = make_linked_ptr(schema); 215 if (!GeneratedSchemas::IsGenerated(schema_namespace)) 216 CHECK_EQ(1u, unloaded_schemas_.erase(schema_namespace)); 217 } 218 } 219 220 ExtensionAPI::ExtensionAPI() : default_configuration_initialized_(false) { 221 } 222 223 ExtensionAPI::~ExtensionAPI() { 224 } 225 226 void ExtensionAPI::InitDefaultConfiguration() { 227 RegisterDependencyProvider( 228 "api", FeatureProvider::GetByName("api")); 229 RegisterDependencyProvider( 230 "manifest", FeatureProvider::GetByName("manifest")); 231 RegisterDependencyProvider( 232 "permission", FeatureProvider::GetByName("permission")); 233 234 // Schemas to be loaded from resources. 235 CHECK(unloaded_schemas_.empty()); 236 RegisterSchemaResource("app", IDR_EXTENSION_API_JSON_APP); 237 RegisterSchemaResource("browserAction", IDR_EXTENSION_API_JSON_BROWSERACTION); 238 RegisterSchemaResource("browsingData", IDR_EXTENSION_API_JSON_BROWSINGDATA); 239 RegisterSchemaResource("commands", IDR_EXTENSION_API_JSON_COMMANDS); 240 RegisterSchemaResource("declarativeContent", 241 IDR_EXTENSION_API_JSON_DECLARATIVE_CONTENT); 242 RegisterSchemaResource("declarativeWebRequest", 243 IDR_EXTENSION_API_JSON_DECLARATIVE_WEBREQUEST); 244 RegisterSchemaResource("experimental.input.virtualKeyboard", 245 IDR_EXTENSION_API_JSON_EXPERIMENTAL_INPUT_VIRTUALKEYBOARD); 246 RegisterSchemaResource("experimental.processes", 247 IDR_EXTENSION_API_JSON_EXPERIMENTAL_PROCESSES); 248 RegisterSchemaResource("experimental.rlz", 249 IDR_EXTENSION_API_JSON_EXPERIMENTAL_RLZ); 250 RegisterSchemaResource("runtime", IDR_EXTENSION_API_JSON_RUNTIME); 251 RegisterSchemaResource("fileBrowserHandler", 252 IDR_EXTENSION_API_JSON_FILEBROWSERHANDLER); 253 RegisterSchemaResource("fileBrowserPrivate", 254 IDR_EXTENSION_API_JSON_FILEBROWSERPRIVATE); 255 RegisterSchemaResource("inputMethodPrivate", 256 IDR_EXTENSION_API_JSON_INPUTMETHODPRIVATE); 257 RegisterSchemaResource("pageAction", IDR_EXTENSION_API_JSON_PAGEACTION); 258 RegisterSchemaResource("pageActions", IDR_EXTENSION_API_JSON_PAGEACTIONS); 259 RegisterSchemaResource("privacy", IDR_EXTENSION_API_JSON_PRIVACY); 260 RegisterSchemaResource("proxy", IDR_EXTENSION_API_JSON_PROXY); 261 RegisterSchemaResource("scriptBadge", IDR_EXTENSION_API_JSON_SCRIPTBADGE); 262 RegisterSchemaResource("streamsPrivate", 263 IDR_EXTENSION_API_JSON_STREAMSPRIVATE); 264 RegisterSchemaResource("ttsEngine", IDR_EXTENSION_API_JSON_TTSENGINE); 265 RegisterSchemaResource("tts", IDR_EXTENSION_API_JSON_TTS); 266 RegisterSchemaResource("types", IDR_EXTENSION_API_JSON_TYPES); 267 RegisterSchemaResource("types.private", IDR_EXTENSION_API_JSON_TYPES_PRIVATE); 268 RegisterSchemaResource("webRequestInternal", 269 IDR_EXTENSION_API_JSON_WEBREQUESTINTERNAL); 270 RegisterSchemaResource("webstore", IDR_EXTENSION_API_JSON_WEBSTORE); 271 RegisterSchemaResource("webstorePrivate", 272 IDR_EXTENSION_API_JSON_WEBSTOREPRIVATE); 273 default_configuration_initialized_ = true; 274 } 275 276 void ExtensionAPI::RegisterSchemaResource(const std::string& name, 277 int resource_id) { 278 unloaded_schemas_[name] = resource_id; 279 } 280 281 void ExtensionAPI::RegisterDependencyProvider(const std::string& name, 282 FeatureProvider* provider) { 283 dependency_providers_[name] = provider; 284 } 285 286 bool ExtensionAPI::IsAnyFeatureAvailableToContext(const std::string& api_name, 287 Feature::Context context, 288 const GURL& url) { 289 FeatureProviderMap::iterator provider = dependency_providers_.find("api"); 290 CHECK(provider != dependency_providers_.end()); 291 const std::vector<std::string>& features = 292 provider->second->GetAllFeatureNames(); 293 294 // Check to see if there are any parts of this API that are allowed in this 295 // context. 296 for (std::vector<std::string>::const_iterator i = features.begin(); 297 i != features.end(); ++i) { 298 const std::string& feature_name = *i; 299 if (feature_name != api_name && feature_name.find(api_name + ".") == 0) { 300 if (IsAvailable(feature_name, NULL, context, url).is_available()) 301 return true; 302 } 303 } 304 return IsAvailable(api_name, NULL, context, url).is_available(); 305 } 306 307 Feature::Availability ExtensionAPI::IsAvailable(const std::string& full_name, 308 const Extension* extension, 309 Feature::Context context, 310 const GURL& url) { 311 Feature* feature = GetFeatureDependency(full_name); 312 CHECK(feature) << full_name; 313 314 Feature::Availability availability = 315 feature->IsAvailableToContext(extension, context, url); 316 if (!availability.is_available()) 317 return availability; 318 319 for (std::set<std::string>::iterator iter = feature->dependencies().begin(); 320 iter != feature->dependencies().end(); ++iter) { 321 Feature::Availability dependency_availability = 322 IsAvailable(*iter, extension, context, url); 323 if (!dependency_availability.is_available()) 324 return dependency_availability; 325 } 326 327 return Feature::CreateAvailability(Feature::IS_AVAILABLE, std::string()); 328 } 329 330 bool ExtensionAPI::IsPrivileged(const std::string& full_name) { 331 Feature* feature = GetFeatureDependency(full_name); 332 CHECK(feature); 333 DCHECK(!feature->GetContexts()->empty()); 334 // An API is 'privileged' if it can only be run in a blessed context. 335 return feature->GetContexts()->size() == 336 feature->GetContexts()->count(Feature::BLESSED_EXTENSION_CONTEXT); 337 } 338 339 const base::DictionaryValue* ExtensionAPI::GetSchema( 340 const std::string& full_name) { 341 std::string child_name; 342 std::string api_name = GetAPINameFromFullName(full_name, &child_name); 343 344 const base::DictionaryValue* result = NULL; 345 SchemaMap::iterator maybe_schema = schemas_.find(api_name); 346 if (maybe_schema != schemas_.end()) { 347 result = maybe_schema->second.get(); 348 } else { 349 // Might not have loaded yet; or might just not exist. 350 UnloadedSchemaMap::iterator maybe_schema_resource = 351 unloaded_schemas_.find(api_name); 352 if (maybe_schema_resource != unloaded_schemas_.end()) { 353 LoadSchema(maybe_schema_resource->first, 354 ReadFromResource(maybe_schema_resource->second)); 355 } else if (default_configuration_initialized_ && 356 GeneratedSchemas::IsGenerated(api_name)) { 357 LoadSchema(api_name, GeneratedSchemas::Get(api_name)); 358 } else { 359 return NULL; 360 } 361 362 maybe_schema = schemas_.find(api_name); 363 CHECK(schemas_.end() != maybe_schema); 364 result = maybe_schema->second.get(); 365 } 366 367 if (!child_name.empty()) 368 result = GetSchemaChild(result, child_name); 369 370 return result; 371 } 372 373 Feature* ExtensionAPI::GetFeatureDependency(const std::string& full_name) { 374 std::string feature_type; 375 std::string feature_name; 376 SplitDependencyName(full_name, &feature_type, &feature_name); 377 378 FeatureProviderMap::iterator provider = 379 dependency_providers_.find(feature_type); 380 if (provider == dependency_providers_.end()) 381 return NULL; 382 383 Feature* feature = provider->second->GetFeature(feature_name); 384 // Try getting the feature for the parent API, if this was a child. 385 if (!feature) { 386 std::string child_name; 387 feature = provider->second->GetFeature( 388 GetAPINameFromFullName(feature_name, &child_name)); 389 } 390 return feature; 391 } 392 393 std::string ExtensionAPI::GetAPINameFromFullName(const std::string& full_name, 394 std::string* child_name) { 395 std::string api_name_candidate = full_name; 396 while (true) { 397 if (schemas_.find(api_name_candidate) != schemas_.end() || 398 GeneratedSchemas::IsGenerated(api_name_candidate) || 399 unloaded_schemas_.find(api_name_candidate) != unloaded_schemas_.end()) { 400 std::string result = api_name_candidate; 401 402 if (child_name) { 403 if (result.length() < full_name.length()) 404 *child_name = full_name.substr(result.length() + 1); 405 else 406 *child_name = ""; 407 } 408 409 return result; 410 } 411 412 size_t last_dot_index = api_name_candidate.rfind('.'); 413 if (last_dot_index == std::string::npos) 414 break; 415 416 api_name_candidate = api_name_candidate.substr(0, last_dot_index); 417 } 418 419 *child_name = ""; 420 return std::string(); 421 } 422 423 } // namespace extensions 424