blob: a4930146acbfa78de1a41a384bb16c0465755aad [file] [log] [blame]
// Copyright (c) 2010 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/plugin_group.h"
#include "base/linked_ptr.h"
#include "base/string_util.h"
#include "base/sys_string_conversions.h"
#include "base/utf_string_conversions.h"
#include "base/values.h"
#include "base/version.h"
#include "webkit/glue/plugins/plugin_list.h"
#include "webkit/glue/plugins/webplugininfo.h"
#if defined(OS_MACOSX)
// Plugin Groups for Mac.
// Plugins are listed here as soon as vulnerabilities and solutions
// (new versions) are published.
// TODO(panayiotis): Get the Real Player version on Mac, somehow.
static const PluginGroupDefinition kGroupDefinitions[] = {
{ "apple-quicktime", "Quicktime", "QuickTime Plug-in", "", "", "7.6.6",
"http://www.apple.com/quicktime/download/" },
{ "java-runtime-environment", "Java", "Java", "", "", "",
"http://support.apple.com/kb/HT1338" },
{ "adobe-flash-player", "Flash", "Shockwave Flash", "", "", "10.1.82",
"http://get.adobe.com/flashplayer/" },
{ "silverlight-3", "Silverlight 3", "Silverlight", "0", "4", "3.0.50106.0",
"http://go.microsoft.com/fwlink/?LinkID=185927" },
{ "silverlight-4", "Silverlight 4", "Silverlight", "4", "5", "",
"http://go.microsoft.com/fwlink/?LinkID=185927" },
{ "flip4mac", "Flip4Mac", "Flip4Mac", "", "", "2.2.1",
"http://www.telestream.net/flip4mac-wmv/overview.htm" },
{ "shockwave", "Shockwave", "Shockwave for Director", "", "", "11.5.8.612",
"http://www.adobe.com/shockwave/download/" }
};
#elif defined(OS_WIN)
// TODO(panayiotis): We should group "RealJukebox NS Plugin" with the rest of
// the RealPlayer files.
static const PluginGroupDefinition kGroupDefinitions[] = {
{ "apple-quicktime", "Quicktime", "QuickTime Plug-in", "", "", "7.6.7",
"http://www.apple.com/quicktime/download/" },
{ "java-runtime-environment", "Java 6", "Java", "", "6", "6.0.200",
"http://www.java.com/" },
{ "adobe-reader", "Adobe Reader 9", "Adobe Acrobat", "9", "10", "9.3.3",
"http://get.adobe.com/reader/" },
{ "adobe-reader-8", "Adobe Reader 8", "Adobe Acrobat", "0", "9", "8.2.3",
"http://get.adobe.com/reader/" },
{ "adobe-flash-player", "Flash", "Shockwave Flash", "", "", "10.1.82",
"http://get.adobe.com/flashplayer/" },
{ "silverlight-3", "Silverlight 3", "Silverlight", "0", "4", "3.0.50106.0",
"http://go.microsoft.com/fwlink/?LinkID=185927" },
{ "silverlight-4", "Silverlight 4", "Silverlight", "4", "5", "",
"http://go.microsoft.com/fwlink/?LinkID=185927" },
{ "shockwave", "Shockwave", "Shockwave for Director", "", "", "11.5.8.612",
"http://www.adobe.com/shockwave/download/" },
{ "divx-player", "DivX Player", "DivX Web Player", "", "", "1.4.3.4",
"http://download.divx.com/divx/autoupdate/player/"
"DivXWebPlayerInstaller.exe" },
// These are here for grouping, no vulnerabilies known.
{ "windows-media-player", "Windows Media Player", "Windows Media Player",
"", "", "", "" },
{ "microsoft-office", "Microsoft Office", "Microsoft Office",
"", "", "", "" },
// TODO(panayiotis): The vulnerable versions are
// (v >= 6.0.12.1040 && v <= 6.0.12.1663)
// || v == 6.0.12.1698 || v == 6.0.12.1741
{ "realplayer", "RealPlayer", "RealPlayer", "", "", "",
"http://www.adobe.com/shockwave/download/" },
};
#else
static const PluginGroupDefinition kGroupDefinitions[] = {};
#endif
/*static*/
std::set<string16>* PluginGroup::policy_disabled_plugin_patterns_;
/*static*/
const PluginGroupDefinition* PluginGroup::GetPluginGroupDefinitions() {
return kGroupDefinitions;
}
/*static*/
size_t PluginGroup::GetPluginGroupDefinitionsSize() {
// TODO(viettrungluu): |arraysize()| doesn't work with zero-size arrays.
return ARRAYSIZE_UNSAFE(kGroupDefinitions);
}
/*static*/
void PluginGroup::SetPolicyDisabledPluginPatterns(
const std::set<string16>& set) {
if (!policy_disabled_plugin_patterns_)
policy_disabled_plugin_patterns_ = new std::set<string16>(set);
else
*policy_disabled_plugin_patterns_ = set;
}
/*static*/
bool PluginGroup::IsPluginNameDisabledByPolicy(const string16& plugin_name) {
if (!policy_disabled_plugin_patterns_)
return false;
std::set<string16>::const_iterator pattern(
policy_disabled_plugin_patterns_->begin());
while (pattern != policy_disabled_plugin_patterns_->end()) {
if (MatchPattern(plugin_name, *pattern))
return true;
++pattern;
}
return false;
}
/*static*/
bool PluginGroup::IsPluginPathDisabledByPolicy(const FilePath& plugin_path) {
std::vector<WebPluginInfo> plugins;
NPAPI::PluginList::Singleton()->GetPlugins(false, &plugins);
for (std::vector<WebPluginInfo>::const_iterator it = plugins.begin();
it != plugins.end();
++it) {
if (FilePath::CompareEqualIgnoreCase(it->path.value(),
plugin_path.value()) && IsPluginNameDisabledByPolicy(it->name)) {
return true;
}
}
return false;
}
PluginGroup::PluginGroup(const string16& group_name,
const string16& name_matcher,
const std::string& version_range_low,
const std::string& version_range_high,
const std::string& min_version,
const std::string& update_url,
const std::string& identifier)
: identifier_(identifier),
group_name_(group_name),
name_matcher_(name_matcher),
version_range_low_str_(version_range_low),
version_range_high_str_(version_range_high),
update_url_(update_url),
enabled_(false),
min_version_str_(min_version),
version_(Version::GetVersionFromString("0")) {
if (!version_range_low.empty())
version_range_low_.reset(Version::GetVersionFromString(version_range_low));
if (!version_range_high.empty()) {
version_range_high_.reset(
Version::GetVersionFromString(version_range_high));
}
if (!min_version.empty())
min_version_.reset(Version::GetVersionFromString(min_version));
}
PluginGroup* PluginGroup::FromPluginGroupDefinition(
const PluginGroupDefinition& definition) {
return new PluginGroup(ASCIIToUTF16(definition.name),
ASCIIToUTF16(definition.name_matcher),
definition.version_matcher_low,
definition.version_matcher_high,
definition.min_version,
definition.update_url,
definition.identifier);
}
PluginGroup::~PluginGroup() { }
PluginGroup* PluginGroup::FromWebPluginInfo(const WebPluginInfo& wpi) {
// Create a matcher from the name of this plugin.
#if defined(OS_POSIX)
std::string identifier = wpi.path.BaseName().value();
#elif defined(OS_WIN)
std::string identifier = base::SysWideToUTF8(wpi.path.BaseName().value());
#endif
return new PluginGroup(wpi.name, wpi.name, std::string(), std::string(),
std::string(), std::string(), identifier);
}
PluginGroup* PluginGroup::CopyOrCreatePluginGroup(
const WebPluginInfo& info) {
static PluginMap* hardcoded_plugin_groups = NULL;
if (!hardcoded_plugin_groups) {
PluginMap* groups = new PluginMap();
const PluginGroupDefinition* definitions = GetPluginGroupDefinitions();
for (size_t i = 0; i < GetPluginGroupDefinitionsSize(); ++i) {
PluginGroup* definition_group = PluginGroup::FromPluginGroupDefinition(
definitions[i]);
std::string identifier = definition_group->identifier();
DCHECK(groups->find(identifier) == groups->end());
(*groups)[identifier] = linked_ptr<PluginGroup>(definition_group);
}
hardcoded_plugin_groups = groups;
}
// See if this plugin matches any of the hardcoded groups.
PluginGroup* hardcoded_group = FindGroupMatchingPlugin(
*hardcoded_plugin_groups, info);
if (hardcoded_group) {
// Make a copy.
return hardcoded_group->Copy();
} else {
// Not found in our hardcoded list, create a new one.
return PluginGroup::FromWebPluginInfo(info);
}
}
PluginGroup* PluginGroup::FindGroupMatchingPlugin(
const PluginMap& plugin_groups,
const WebPluginInfo& plugin) {
for (std::map<std::string, linked_ptr<PluginGroup> >::const_iterator it =
plugin_groups.begin();
it != plugin_groups.end();
++it) {
if (it->second->Match(plugin))
return it->second.get();
}
return NULL;
}
bool PluginGroup::Match(const WebPluginInfo& plugin) const {
if (name_matcher_.empty()) {
return false;
}
// Look for the name matcher anywhere in the plugin name.
if (plugin.name.find(name_matcher_) == string16::npos) {
return false;
}
if (version_range_low_.get() == NULL ||
version_range_high_.get() == NULL) {
return true;
}
// There's a version range, we must be in it.
scoped_ptr<Version> plugin_version(
Version::GetVersionFromString(UTF16ToWide(plugin.version)));
if (plugin_version.get() == NULL) {
// No version could be extracted, assume we don't match the range.
return false;
}
// We match if we are in the range: [low, high)
return (version_range_low_->CompareTo(*plugin_version) <= 0 &&
version_range_high_->CompareTo(*plugin_version) > 0);
}
Version* PluginGroup::CreateVersionFromString(const string16& version_string) {
// Remove spaces and ')' from the version string,
// Replace any instances of 'r', ',' or '(' with a dot.
std::wstring version = UTF16ToWide(version_string);
RemoveChars(version, L") ", &version);
std::replace(version.begin(), version.end(), 'r', '.');
std::replace(version.begin(), version.end(), ',', '.');
std::replace(version.begin(), version.end(), '(', '.');
return Version::GetVersionFromString(version);
}
void PluginGroup::UpdateActivePlugin(const WebPluginInfo& plugin) {
// A group is enabled if any of the files are enabled.
if (plugin.enabled) {
if (!enabled_) {
// If this is the first enabled plugin, use its description.
enabled_ = true;
UpdateDescriptionAndVersion(plugin);
}
} else {
// If this is the first plugin and it's disabled,
// use its description for now.
if (description_.empty())
UpdateDescriptionAndVersion(plugin);
}
}
void PluginGroup::UpdateDescriptionAndVersion(const WebPluginInfo& plugin) {
description_ = plugin.desc;
if (Version* new_version = CreateVersionFromString(plugin.version))
version_.reset(new_version);
else
version_.reset(Version::GetVersionFromString("0"));
}
void PluginGroup::AddPlugin(const WebPluginInfo& plugin, int position) {
web_plugin_infos_.push_back(plugin);
// The position of this plugin relative to the global list of plugins.
web_plugin_positions_.push_back(position);
UpdateActivePlugin(plugin);
}
string16 PluginGroup::GetGroupName() const {
if (!group_name_.empty())
return group_name_;
DCHECK_EQ(1u, web_plugin_infos_.size());
FilePath::StringType path =
web_plugin_infos_[0].path.BaseName().RemoveExtension().value();
#if defined(OS_POSIX)
return UTF8ToUTF16(path);
#elif defined(OS_WIN)
return WideToUTF16(path);
#endif
}
DictionaryValue* PluginGroup::GetSummary() const {
DictionaryValue* result = new DictionaryValue();
result->SetString("name", GetGroupName());
result->SetBoolean("enabled", enabled_);
return result;
}
DictionaryValue* PluginGroup::GetDataForUI() const {
string16 name = GetGroupName();
DictionaryValue* result = new DictionaryValue();
result->SetString("name", name);
result->SetString("description", description_);
result->SetString("version", version_->GetString());
result->SetString("update_url", update_url_);
result->SetBoolean("critical", IsVulnerable());
bool group_disabled_by_policy = IsPluginNameDisabledByPolicy(name);
ListValue* plugin_files = new ListValue();
bool all_plugins_disabled_by_policy = true;
for (size_t i = 0; i < web_plugin_infos_.size(); ++i) {
const WebPluginInfo& web_plugin = web_plugin_infos_[i];
int priority = web_plugin_positions_[i];
DictionaryValue* plugin_file = new DictionaryValue();
plugin_file->SetString("name", web_plugin.name);
plugin_file->SetString("description", web_plugin.desc);
plugin_file->SetString("path", web_plugin.path.value());
plugin_file->SetString("version", web_plugin.version);
bool plugin_disabled_by_policy = group_disabled_by_policy ||
IsPluginNameDisabledByPolicy(web_plugin.name);
if (plugin_disabled_by_policy) {
plugin_file->SetString("enabledMode", "disabledByPolicy");
} else {
all_plugins_disabled_by_policy = false;
plugin_file->SetString("enabledMode",
web_plugin.enabled ? "enabled" : "disabledByUser");
}
plugin_file->SetInteger("priority", priority);
ListValue* mime_types = new ListValue();
for (std::vector<WebPluginMimeType>::const_iterator type_it =
web_plugin.mime_types.begin();
type_it != web_plugin.mime_types.end();
++type_it) {
DictionaryValue* mime_type = new DictionaryValue();
mime_type->SetString("mimeType", type_it->mime_type);
mime_type->SetString("description", type_it->description);
ListValue* file_extensions = new ListValue();
for (std::vector<std::string>::const_iterator ext_it =
type_it->file_extensions.begin();
ext_it != type_it->file_extensions.end();
++ext_it) {
file_extensions->Append(new StringValue(*ext_it));
}
mime_type->Set("fileExtensions", file_extensions);
mime_types->Append(mime_type);
}
plugin_file->Set("mimeTypes", mime_types);
plugin_files->Append(plugin_file);
}
if (group_disabled_by_policy || all_plugins_disabled_by_policy) {
result->SetString("enabledMode", "disabledByPolicy");
} else {
result->SetString("enabledMode", enabled_ ? "enabled" : "disabledByUser");
}
result->Set("plugin_files", plugin_files);
return result;
}
// Returns true if the latest version of this plugin group is vulnerable.
bool PluginGroup::IsVulnerable() const {
if (min_version_.get() == NULL || version_->GetString() == "0") {
return false;
}
return version_->CompareTo(*min_version_) < 0;
}
void PluginGroup::DisableOutdatedPlugins() {
if (!min_version_.get())
return;
description_ = string16();
enabled_ = false;
for (std::vector<WebPluginInfo>::iterator it =
web_plugin_infos_.begin();
it != web_plugin_infos_.end(); ++it) {
scoped_ptr<Version> version(CreateVersionFromString(it->version));
if (version.get() && version->CompareTo(*min_version_) < 0) {
it->enabled = false;
NPAPI::PluginList::Singleton()->DisablePlugin(it->path);
}
UpdateActivePlugin(*it);
}
}
void PluginGroup::Enable(bool enable) {
for (std::vector<WebPluginInfo>::const_iterator it =
web_plugin_infos_.begin();
it != web_plugin_infos_.end(); ++it) {
if (enable && !IsPluginNameDisabledByPolicy(it->name)) {
NPAPI::PluginList::Singleton()->EnablePlugin(it->path);
} else {
NPAPI::PluginList::Singleton()->DisablePlugin(it->path);
}
}
}