// Copyright 2014 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/ui/pdf/adobe_reader_info_win.h" #include #include #include #include "base/bind.h" #include "base/callback.h" #include "base/file_version_info.h" #include "base/strings/string_util.h" #include "base/strings/stringprintf.h" #include "base/strings/utf_string_conversions.h" #include "base/win/registry.h" #include "base/win/windows_version.h" #include "chrome/browser/browser_process.h" #include "chrome/browser/plugins/plugin_finder.h" #include "chrome/browser/plugins/plugin_metadata.h" #include "chrome/browser/plugins/plugin_prefs.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/profiles/profile_manager.h" #include "content/public/browser/plugin_service.h" namespace { // Hardcoded value for the secure version of Acrobat Reader. const char kSecureVersion[] = "11.0.8.4"; const char kAdobeReaderIdentifier[] = "adobe-reader"; const char kPdfMimeType[] = "application/pdf"; const base::char16 kRegistryAcrobat[] = L"Acrobat.exe"; const base::char16 kRegistryAcrobatReader[] = L"AcroRd32.exe"; const base::char16 kRegistryApps[] = L"Software\\Microsoft\\Windows\\CurrentVersion\\App Paths"; const base::char16 kRegistryPath[] = L"Path"; // Gets the installed path for a registered app. base::FilePath GetInstalledPath(const base::char16* app) { base::string16 reg_path(kRegistryApps); reg_path.append(L"\\"); reg_path.append(app); base::FilePath filepath; base::win::RegKey hkcu_key(HKEY_CURRENT_USER, reg_path.c_str(), KEY_READ); base::string16 path; // As of Win7 AppPaths can also be registered in HKCU: http://goo.gl/UgFOf. if (base::win::GetVersion() >= base::win::VERSION_WIN7 && hkcu_key.ReadValue(kRegistryPath, &path) == ERROR_SUCCESS) { filepath = base::FilePath(path); } else { base::win::RegKey hklm_key(HKEY_LOCAL_MACHINE, reg_path.c_str(), KEY_READ); if (hklm_key.ReadValue(kRegistryPath, &path) == ERROR_SUCCESS) { filepath = base::FilePath(path); } } return filepath.Append(app); } bool IsPdfMimeType(const content::WebPluginMimeType& plugin_mime_type) { return plugin_mime_type.mime_type == kPdfMimeType; } AdobeReaderPluginInfo GetReaderPlugin( Profile* profile, const std::vector& plugins) { AdobeReaderPluginInfo reader_info; reader_info.is_installed = false; reader_info.is_enabled = false; reader_info.is_secure = false; PluginFinder* plugin_finder = PluginFinder::GetInstance(); for (size_t i = 0; i < plugins.size(); ++i) { const content::WebPluginInfo& plugin = plugins[i]; if (plugin.is_pepper_plugin()) continue; if (std::find_if(plugin.mime_types.begin(), plugin.mime_types.end(), IsPdfMimeType) == plugin.mime_types.end()) continue; scoped_ptr plugin_metadata( plugin_finder->GetPluginMetadata(plugins[i])); if (plugin_metadata->identifier() != kAdobeReaderIdentifier) continue; reader_info.is_installed = true; if (profile) { PluginPrefs* plugin_prefs = PluginPrefs::GetForProfile(profile); PluginPrefs::PolicyStatus plugin_status = plugin_prefs->PolicyStatusForPlugin(plugin_metadata->name()); reader_info.is_enabled = plugin_status != PluginPrefs::POLICY_DISABLED; } // Adobe Reader will likely always come up as "requires_authorization". // See http://crbug.com/311655. PluginMetadata::SecurityStatus security_stat = plugin_metadata->GetSecurityStatus(plugins[i]); reader_info.is_secure = security_stat == PluginMetadata::SECURITY_STATUS_UP_TO_DATE || security_stat == PluginMetadata::SECURITY_STATUS_REQUIRES_AUTHORIZATION; reader_info.plugin_info = plugins[i]; break; } return reader_info; } void OnGotPluginInfo(Profile* profile, const GetAdobeReaderPluginInfoCallback& callback, const std::vector& plugins) { if (!g_browser_process->profile_manager()->IsValidProfile(profile)) profile = NULL; callback.Run(GetReaderPlugin(profile, plugins)); } bool IsAdobeReaderDefaultPDFViewerInternal(base::FilePath* path) { base::char16 app_cmd_buf[MAX_PATH]; DWORD app_cmd_buf_len = MAX_PATH; HRESULT hr = AssocQueryString(ASSOCF_NONE, ASSOCSTR_COMMAND, L".pdf", L"open", app_cmd_buf, &app_cmd_buf_len); if (FAILED(hr)) return false; // Looks for the install paths for Acrobat / Reader. base::FilePath install_path = GetInstalledPath(kRegistryAcrobatReader); if (install_path.empty()) install_path = GetInstalledPath(kRegistryAcrobat); if (install_path.empty()) return false; base::string16 app_cmd(app_cmd_buf); bool found = app_cmd.find(install_path.value()) != base::string16::npos; if (found && path) *path = install_path; return found; } } // namespace void GetAdobeReaderPluginInfoAsync( Profile* profile, const GetAdobeReaderPluginInfoCallback& callback) { DCHECK(!callback.is_null()); content::PluginService::GetInstance()->GetPlugins( base::Bind(&OnGotPluginInfo, profile, callback)); } bool GetAdobeReaderPluginInfo(Profile* profile, AdobeReaderPluginInfo* reader_info) { DCHECK(reader_info); std::vector plugins; bool up_to_date = content::PluginService::GetInstance()->GetPluginInfoArray( GURL(), kPdfMimeType, false, &plugins, NULL); *reader_info = GetReaderPlugin(profile, plugins); return up_to_date; } bool IsAdobeReaderDefaultPDFViewer() { return IsAdobeReaderDefaultPDFViewerInternal(NULL); } bool IsAdobeReaderUpToDate() { base::FilePath install_path; bool is_default = IsAdobeReaderDefaultPDFViewerInternal(&install_path); if (!is_default) return false; scoped_ptr file_version_info( FileVersionInfo::CreateFileVersionInfo(install_path)); if (!file_version_info) return false; std::string reader_version = base::UTF16ToUTF8(file_version_info->product_version()); // Convert 1.2.03.45 to 1.2.3.45 so base::Version considers it as valid. for (int i = 1; i <= 9; ++i) { std::string from = base::StringPrintf(".0%d", i); std::string to = base::StringPrintf(".%d", i); ReplaceSubstringsAfterOffset(&reader_version, 0, from, to); } base::Version file_version(reader_version); return file_version.IsValid() && !file_version.IsOlderThan(kSecureVersion); }