// 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/browser/plugin_finder.h" #include "base/bind.h" #include "base/json/json_reader.h" #include "base/message_loop.h" #include "base/stl_util.h" #include "base/values.h" #include "chrome/browser/browser_process.h" #include "chrome/browser/plugin_installer.h" #include "chrome/browser/prefs/pref_service.h" #include "chrome/common/pref_names.h" #include "content/public/browser/browser_thread.h" #include "googleurl/src/gurl.h" #include "grit/browser_resources.h" #include "ui/base/resource/resource_bundle.h" // static PluginFinder* PluginFinder::GetInstance() { return Singleton::get(); } PluginFinder::PluginFinder() : plugin_list_(LoadPluginList()) { if (!plugin_list_.get()) { NOTREACHED(); plugin_list_.reset(new base::ListValue()); } } // static scoped_ptr PluginFinder::LoadPluginList() { return scoped_ptr(LoadPluginListInternal()).Pass(); } base::ListValue* PluginFinder::LoadPluginListInternal() { #if defined(OS_WIN) || defined(OS_MACOSX) || defined(OS_LINUX) base::StringPiece json_resource( ResourceBundle::GetSharedInstance().GetRawDataResource( IDR_PLUGIN_DB_JSON)); bool allow_trailing_comma = false; std::string error_str; scoped_ptr value(base::JSONReader::ReadAndReturnError( json_resource.as_string(), allow_trailing_comma, NULL, &error_str)); if (!value.get()) { DLOG(ERROR) << error_str; return NULL; } base::DictionaryValue* dict = NULL; if (!value->GetAsDictionary(&dict)) return NULL; base::ListValue* list = NULL; if (!dict->GetList("plugins", &list)) return NULL; return list->DeepCopy(); #else return new base::ListValue(); #endif } PluginFinder::~PluginFinder() { STLDeleteValues(&installers_); } void PluginFinder::FindPlugin( const std::string& mime_type, const std::string& language, const FindPluginCallback& callback) { PluginInstaller* installer = FindPluginInternal(mime_type, language); MessageLoop::current()->PostTask(FROM_HERE, base::Bind(callback, installer)); } void PluginFinder::FindPluginWithIdentifier( const std::string& identifier, const FindPluginCallback& found_callback) { PluginInstaller* installer = NULL; std::map::const_iterator it = installers_.find(identifier); if (it != installers_.end()) { installer = it->second; } else { for (ListValue::const_iterator plugin_it = plugin_list_->begin(); plugin_it != plugin_list_->end(); ++plugin_it) { const base::DictionaryValue* plugin = NULL; if (!(*plugin_it)->GetAsDictionary(&plugin)) { NOTREACHED(); continue; } std::string id; bool success = plugin->GetString("identifier", &id); DCHECK(success); if (id == identifier) { installer = CreateInstaller(identifier, plugin); break; } } } MessageLoop::current()->PostTask(FROM_HERE, base::Bind(found_callback, installer)); } PluginInstaller* PluginFinder::CreateInstaller( const std::string& identifier, const base::DictionaryValue* plugin_dict) { DCHECK(!installers_[identifier]); std::string url; bool success = plugin_dict->GetString("url", &url); DCHECK(success); std::string help_url; plugin_dict->GetString("help_url", &help_url); string16 name; success = plugin_dict->GetString("name", &name); DCHECK(success); bool display_url = false; plugin_dict->GetBoolean("displayurl", &display_url); PluginInstaller*installer = new PluginInstaller(identifier, GURL(url), GURL(help_url), name, display_url); installers_[identifier] = installer; return installer; } PluginInstaller* PluginFinder::FindPluginInternal( const std::string& mime_type, const std::string& language) { if (!g_browser_process->local_state()->GetBoolean( prefs::kDisablePluginFinder)) { for (ListValue::const_iterator plugin_it = plugin_list_->begin(); plugin_it != plugin_list_->end(); ++plugin_it) { const base::DictionaryValue* plugin = NULL; if (!(*plugin_it)->GetAsDictionary(&plugin)) { NOTREACHED(); continue; } std::string language_str; bool success = plugin->GetString("lang", &language_str); DCHECK(success); if (language_str != language) continue; ListValue* mime_types = NULL; success = plugin->GetList("mime_types", &mime_types); DCHECK(success); for (ListValue::const_iterator mime_type_it = mime_types->begin(); mime_type_it != mime_types->end(); ++mime_type_it) { std::string mime_type_str; success = (*mime_type_it)->GetAsString(&mime_type_str); DCHECK(success); if (mime_type_str == mime_type) { std::string identifier; bool success = plugin->GetString("identifier", &identifier); DCHECK(success); std::map::const_iterator it = installers_.find(identifier); if (it != installers_.end()) return it->second; return CreateInstaller(identifier, plugin); } } } } return NULL; }