// 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/download/download_prefs.h" #include #include #include #include "base/bind.h" #include "base/bind_helpers.h" #include "base/files/file_util.h" #include "base/lazy_instance.h" #include "base/logging.h" #include "base/macros.h" #include "base/path_service.h" #include "base/strings/string_split.h" #include "base/strings/string_util.h" #include "base/strings/sys_string_conversions.h" #include "base/strings/utf_string_conversions.h" #include "build/build_config.h" #include "chrome/browser/download/chrome_download_manager_delegate.h" #include "chrome/browser/download/download_extensions.h" #include "chrome/browser/download/download_service.h" #include "chrome/browser/download/download_service_factory.h" #include "chrome/browser/download/download_target_determiner.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/profiles/profile_manager.h" #include "chrome/common/chrome_paths.h" #include "chrome/common/pref_names.h" #include "components/pref_registry/pref_registry_syncable.h" #include "components/prefs/pref_service.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/download_manager.h" #include "content/public/browser/save_page_type.h" #if defined(OS_CHROMEOS) #include "chrome/browser/chromeos/drive/drive_integration_service.h" #include "chrome/browser/chromeos/drive/file_system_util.h" #include "chrome/browser/chromeos/file_manager/path_util.h" #endif #if defined(OS_WIN) #include "chrome/browser/ui/pdf/adobe_reader_info_win.h" #endif using content::BrowserContext; using content::BrowserThread; using content::DownloadManager; namespace { // Consider downloads 'dangerous' if they go to the home directory on Linux and // to the desktop on any platform. bool DownloadPathIsDangerous(const base::FilePath& download_path) { #if defined(OS_LINUX) base::FilePath home_dir = base::GetHomeDir(); if (download_path == home_dir) { return true; } #endif #if defined(OS_ANDROID) // Android does not have a desktop dir. return false; #else base::FilePath desktop_dir; if (!PathService::Get(base::DIR_USER_DESKTOP, &desktop_dir)) { NOTREACHED(); return false; } return (download_path == desktop_dir); #endif } class DefaultDownloadDirectory { public: const base::FilePath& path() const { return path_; } private: friend struct base::DefaultLazyInstanceTraits; DefaultDownloadDirectory() { if (!PathService::Get(chrome::DIR_DEFAULT_DOWNLOADS, &path_)) { NOTREACHED(); } if (DownloadPathIsDangerous(path_)) { // This is only useful on platforms that support // DIR_DEFAULT_DOWNLOADS_SAFE. if (!PathService::Get(chrome::DIR_DEFAULT_DOWNLOADS_SAFE, &path_)) { NOTREACHED(); } } } base::FilePath path_; DISALLOW_COPY_AND_ASSIGN(DefaultDownloadDirectory); }; base::LazyInstance g_default_download_directory = LAZY_INSTANCE_INITIALIZER; } // namespace DownloadPrefs::DownloadPrefs(Profile* profile) : profile_(profile) { PrefService* prefs = profile->GetPrefs(); #if defined(OS_CHROMEOS) // On Chrome OS, the default download directory is different for each profile. // If the profile-unaware default path (from GetDefaultDownloadDirectory()) // is set (this happens during the initial preference registration in static // RegisterProfilePrefs()), alter by GetDefaultDownloadDirectoryForProfile(). // file_manager::util::MigratePathFromOldFormat will do this. const char* path_pref[] = { prefs::kSaveFileDefaultDirectory, prefs::kDownloadDefaultDirectory }; for (size_t i = 0; i < arraysize(path_pref); ++i) { const base::FilePath current = prefs->GetFilePath(path_pref[i]); base::FilePath migrated; if (!current.empty() && file_manager::util::MigratePathFromOldFormat( profile_, current, &migrated)) { prefs->SetFilePath(path_pref[i], migrated); } } // Ensure that the default download directory exists. BrowserThread::PostTask( BrowserThread::FILE, FROM_HERE, base::Bind(base::IgnoreResult(&base::CreateDirectory), GetDefaultDownloadDirectoryForProfile())); #endif // defined(OS_CHROMEOS) #if defined(OS_WIN) || defined(OS_LINUX) || \ (defined(OS_MACOSX) && !defined(OS_IOS)) should_open_pdf_in_system_reader_ = prefs->GetBoolean(prefs::kOpenPdfDownloadInSystemReader); #endif // If the download path is dangerous we forcefully reset it. But if we do // so we set a flag to make sure we only do it once, to avoid fighting // the user if he really wants it on an unsafe place such as the desktop. if (!prefs->GetBoolean(prefs::kDownloadDirUpgraded)) { base::FilePath current_download_dir = prefs->GetFilePath( prefs::kDownloadDefaultDirectory); if (DownloadPathIsDangerous(current_download_dir)) { prefs->SetFilePath(prefs::kDownloadDefaultDirectory, GetDefaultDownloadDirectoryForProfile()); } prefs->SetBoolean(prefs::kDownloadDirUpgraded, true); } prompt_for_download_.Init(prefs::kPromptForDownload, prefs); download_path_.Init(prefs::kDownloadDefaultDirectory, prefs); save_file_path_.Init(prefs::kSaveFileDefaultDirectory, prefs); save_file_type_.Init(prefs::kSaveFileType, prefs); // We store any file extension that should be opened automatically at // download completion in this pref. std::string extensions_to_open = prefs->GetString(prefs::kDownloadExtensionsToOpen); for (const auto& extension_string : base::SplitString( extensions_to_open, ":", base::TRIM_WHITESPACE, base::SPLIT_WANT_ALL)) { #if defined(OS_POSIX) base::FilePath::StringType extension = extension_string; #elif defined(OS_WIN) base::FilePath::StringType extension = base::UTF8ToWide(extension_string); #endif // If it's empty or malformed or not allowed to open automatically, then // skip the entry. Any such entries will be dropped from preferences the // next time SaveAutoOpenState() is called. if (extension.empty() || *extension.begin() == base::FilePath::kExtensionSeparator) continue; // Construct something like ".", since // IsAllowedToOpenAutomatically() needs a filename. base::FilePath filename_with_extension = base::FilePath( base::FilePath::StringType(1, base::FilePath::kExtensionSeparator) + extension); // Note that the list of file types that are not allowed to open // automatically can change in the future. When the list is tightened, it is // expected that some entries in the users' auto open list will get dropped // permanently as a result. if (download_util::IsAllowedToOpenAutomatically(filename_with_extension)) auto_open_.insert(extension); } } DownloadPrefs::~DownloadPrefs() {} // static void DownloadPrefs::RegisterProfilePrefs( user_prefs::PrefRegistrySyncable* registry) { registry->RegisterBooleanPref( prefs::kPromptForDownload, false, user_prefs::PrefRegistrySyncable::SYNCABLE_PREF); registry->RegisterStringPref(prefs::kDownloadExtensionsToOpen, std::string()); registry->RegisterBooleanPref(prefs::kDownloadDirUpgraded, false); registry->RegisterIntegerPref(prefs::kSaveFileType, content::SAVE_PAGE_TYPE_AS_COMPLETE_HTML); const base::FilePath& default_download_path = GetDefaultDownloadDirectory(); registry->RegisterFilePathPref(prefs::kDownloadDefaultDirectory, default_download_path); registry->RegisterFilePathPref(prefs::kSaveFileDefaultDirectory, default_download_path); #if defined(OS_WIN) || defined(OS_LINUX) || \ (defined(OS_MACOSX) && !defined(OS_IOS)) registry->RegisterBooleanPref(prefs::kOpenPdfDownloadInSystemReader, false); #endif } base::FilePath DownloadPrefs::GetDefaultDownloadDirectoryForProfile() const { #if defined(OS_CHROMEOS) return file_manager::util::GetDownloadsFolderForProfile(profile_); #else return GetDefaultDownloadDirectory(); #endif } // static const base::FilePath& DownloadPrefs::GetDefaultDownloadDirectory() { return g_default_download_directory.Get().path(); } // static DownloadPrefs* DownloadPrefs::FromDownloadManager( DownloadManager* download_manager) { ChromeDownloadManagerDelegate* delegate = static_cast( download_manager->GetDelegate()); return delegate->download_prefs(); } // static DownloadPrefs* DownloadPrefs::FromBrowserContext( content::BrowserContext* context) { return FromDownloadManager(BrowserContext::GetDownloadManager(context)); } base::FilePath DownloadPrefs::DownloadPath() const { #if defined(OS_CHROMEOS) // If the download path is under /drive, and DriveIntegrationService isn't // available (which it isn't for incognito mode, for instance), use the // default download directory (/Downloads). if (drive::util::IsUnderDriveMountPoint(*download_path_)) { drive::DriveIntegrationService* integration_service = drive::DriveIntegrationServiceFactory::FindForProfile(profile_); if (!integration_service || !integration_service->is_enabled()) return GetDefaultDownloadDirectoryForProfile(); } #endif return *download_path_; } void DownloadPrefs::SetDownloadPath(const base::FilePath& path) { download_path_.SetValue(path); SetSaveFilePath(path); } base::FilePath DownloadPrefs::SaveFilePath() const { return *save_file_path_; } void DownloadPrefs::SetSaveFilePath(const base::FilePath& path) { save_file_path_.SetValue(path); } void DownloadPrefs::SetSaveFileType(int type) { save_file_type_.SetValue(type); } bool DownloadPrefs::PromptForDownload() const { // If the DownloadDirectory policy is set, then |prompt_for_download_| should // always be false. DCHECK(!download_path_.IsManaged() || !prompt_for_download_.GetValue()); return *prompt_for_download_; } bool DownloadPrefs::IsDownloadPathManaged() const { return download_path_.IsManaged(); } bool DownloadPrefs::IsAutoOpenUsed() const { #if defined(OS_WIN) || defined(OS_LINUX) || \ (defined(OS_MACOSX) && !defined(OS_IOS)) if (ShouldOpenPdfInSystemReader()) return true; #endif return !auto_open_.empty(); } bool DownloadPrefs::IsAutoOpenEnabledBasedOnExtension( const base::FilePath& path) const { base::FilePath::StringType extension = path.Extension(); if (extension.empty()) return false; DCHECK(extension[0] == base::FilePath::kExtensionSeparator); extension.erase(0, 1); #if defined(OS_WIN) || defined(OS_LINUX) || \ (defined(OS_MACOSX) && !defined(OS_IOS)) if (extension == FILE_PATH_LITERAL("pdf") && ShouldOpenPdfInSystemReader()) return true; #endif return auto_open_.find(extension) != auto_open_.end(); } bool DownloadPrefs::EnableAutoOpenBasedOnExtension( const base::FilePath& file_name) { base::FilePath::StringType extension = file_name.Extension(); if (!download_util::IsAllowedToOpenAutomatically(file_name)) return false; DCHECK(extension[0] == base::FilePath::kExtensionSeparator); extension.erase(0, 1); auto_open_.insert(extension); SaveAutoOpenState(); return true; } void DownloadPrefs::DisableAutoOpenBasedOnExtension( const base::FilePath& file_name) { base::FilePath::StringType extension = file_name.Extension(); if (extension.empty()) return; DCHECK(extension[0] == base::FilePath::kExtensionSeparator); extension.erase(0, 1); auto_open_.erase(extension); SaveAutoOpenState(); } #if defined(OS_WIN) || defined(OS_LINUX) || \ (defined(OS_MACOSX) && !defined(OS_IOS)) void DownloadPrefs::SetShouldOpenPdfInSystemReader(bool should_open) { if (should_open_pdf_in_system_reader_ == should_open) return; should_open_pdf_in_system_reader_ = should_open; profile_->GetPrefs()->SetBoolean(prefs::kOpenPdfDownloadInSystemReader, should_open); } bool DownloadPrefs::ShouldOpenPdfInSystemReader() const { #if defined(OS_WIN) if (IsAdobeReaderDefaultPDFViewer() && !DownloadTargetDeterminer::IsAdobeReaderUpToDate()) { return false; } #endif return should_open_pdf_in_system_reader_; } #endif void DownloadPrefs::ResetAutoOpen() { #if defined(OS_WIN) || defined(OS_LINUX) || \ (defined(OS_MACOSX) && !defined(OS_IOS)) SetShouldOpenPdfInSystemReader(false); #endif auto_open_.clear(); SaveAutoOpenState(); } void DownloadPrefs::SaveAutoOpenState() { std::string extensions; for (AutoOpenSet::iterator it = auto_open_.begin(); it != auto_open_.end(); ++it) { #if defined(OS_POSIX) std::string this_extension = *it; #elif defined(OS_WIN) // TODO(phajdan.jr): Why we're using Sys conversion here, but not in ctor? std::string this_extension = base::SysWideToUTF8(*it); #endif extensions += this_extension + ":"; } if (!extensions.empty()) extensions.erase(extensions.size() - 1); profile_->GetPrefs()->SetString(prefs::kDownloadExtensionsToOpen, extensions); } bool DownloadPrefs::AutoOpenCompareFunctor::operator()( const base::FilePath::StringType& a, const base::FilePath::StringType& b) const { return base::FilePath::CompareLessIgnoreCase(a, b); }