328 lines
11 KiB
C++
328 lines
11 KiB
C++
// Copyright (c) 2013 GitHub, Inc.
|
|
// Use of this source code is governed by the MIT license that can be
|
|
// found in the LICENSE file.
|
|
|
|
#include "atom/browser/atom_browser_context.h"
|
|
|
|
#include <utility>
|
|
|
|
#include "atom/browser/atom_blob_reader.h"
|
|
#include "atom/browser/atom_browser_main_parts.h"
|
|
#include "atom/browser/atom_download_manager_delegate.h"
|
|
#include "atom/browser/atom_paths.h"
|
|
#include "atom/browser/atom_permission_manager.h"
|
|
#include "atom/browser/browser.h"
|
|
#include "atom/browser/cookie_change_notifier.h"
|
|
#include "atom/browser/net/resolve_proxy_helper.h"
|
|
#include "atom/browser/pref_store_delegate.h"
|
|
#include "atom/browser/special_storage_policy.h"
|
|
#include "atom/browser/ui/inspectable_web_contents_impl.h"
|
|
#include "atom/browser/web_view_manager.h"
|
|
#include "atom/browser/zoom_level_delegate.h"
|
|
#include "atom/common/application_info.h"
|
|
#include "atom/common/atom_version.h"
|
|
#include "atom/common/chrome_version.h"
|
|
#include "atom/common/options_switches.h"
|
|
#include "base/command_line.h"
|
|
#include "base/files/file_path.h"
|
|
#include "base/path_service.h"
|
|
#include "base/strings/string_util.h"
|
|
#include "base/strings/stringprintf.h"
|
|
#include "base/threading/sequenced_task_runner_handle.h"
|
|
#include "base/threading/thread_restrictions.h"
|
|
#include "chrome/common/chrome_paths.h"
|
|
#include "chrome/common/pref_names.h"
|
|
#include "components/keyed_service/content/browser_context_dependency_manager.h"
|
|
#include "components/prefs/json_pref_store.h"
|
|
#include "components/prefs/pref_registry_simple.h"
|
|
#include "components/prefs/pref_service.h"
|
|
#include "components/prefs/pref_service_factory.h"
|
|
#include "components/prefs/value_map_pref_store.h"
|
|
#include "components/proxy_config/pref_proxy_config_tracker_impl.h"
|
|
#include "components/proxy_config/proxy_config_pref_names.h"
|
|
#include "content/browser/blob_storage/chrome_blob_storage_context.h"
|
|
#include "content/public/browser/storage_partition.h"
|
|
#include "content/public/common/user_agent.h"
|
|
#include "net/base/escape.h"
|
|
|
|
using content::BrowserThread;
|
|
|
|
namespace atom {
|
|
|
|
namespace {
|
|
|
|
std::string RemoveWhitespace(const std::string& str) {
|
|
std::string trimmed;
|
|
if (base::RemoveChars(str, " ", &trimmed))
|
|
return trimmed;
|
|
else
|
|
return str;
|
|
}
|
|
|
|
// Convert string to lower case and escape it.
|
|
std::string MakePartitionName(const std::string& input) {
|
|
return net::EscapePath(base::ToLowerASCII(input));
|
|
}
|
|
|
|
} // namespace
|
|
|
|
// static
|
|
AtomBrowserContext::BrowserContextMap AtomBrowserContext::browser_context_map_;
|
|
|
|
AtomBrowserContext::AtomBrowserContext(const std::string& partition,
|
|
bool in_memory,
|
|
const base::DictionaryValue& options)
|
|
: base::RefCountedDeleteOnSequence<AtomBrowserContext>(
|
|
base::ThreadTaskRunnerHandle::Get()),
|
|
in_memory_pref_store_(nullptr),
|
|
storage_policy_(new SpecialStoragePolicy),
|
|
in_memory_(in_memory),
|
|
weak_factory_(this) {
|
|
// Construct user agent string.
|
|
Browser* browser = Browser::Get();
|
|
std::string name = RemoveWhitespace(browser->GetName());
|
|
std::string user_agent;
|
|
if (name == ATOM_PRODUCT_NAME) {
|
|
user_agent = "Chrome/" CHROME_VERSION_STRING " " ATOM_PRODUCT_NAME
|
|
"/" ATOM_VERSION_STRING;
|
|
} else {
|
|
user_agent = base::StringPrintf(
|
|
"%s/%s Chrome/%s " ATOM_PRODUCT_NAME "/" ATOM_VERSION_STRING,
|
|
name.c_str(), browser->GetVersion().c_str(), CHROME_VERSION_STRING);
|
|
}
|
|
user_agent_ = content::BuildUserAgentFromProduct(user_agent);
|
|
|
|
// Read options.
|
|
base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
|
|
use_cache_ = !command_line->HasSwitch(switches::kDisableHttpCache);
|
|
options.GetBoolean("cache", &use_cache_);
|
|
|
|
base::StringToInt(command_line->GetSwitchValueASCII(switches::kDiskCacheSize),
|
|
&max_cache_size_);
|
|
|
|
if (!base::PathService::Get(DIR_USER_DATA, &path_)) {
|
|
base::PathService::Get(DIR_APP_DATA, &path_);
|
|
path_ = path_.Append(base::FilePath::FromUTF8Unsafe(GetApplicationName()));
|
|
base::PathService::Override(DIR_USER_DATA, path_);
|
|
}
|
|
|
|
if (!in_memory && !partition.empty())
|
|
path_ = path_.Append(FILE_PATH_LITERAL("Partitions"))
|
|
.Append(base::FilePath::FromUTF8Unsafe(
|
|
MakePartitionName(partition)));
|
|
|
|
content::BrowserContext::Initialize(this, path_);
|
|
|
|
// Initialize Pref Registry.
|
|
InitPrefs();
|
|
|
|
proxy_config_monitor_ = std::make_unique<ProxyConfigMonitor>(prefs_.get());
|
|
io_handle_ = new URLRequestContextGetter::Handle(weak_factory_.GetWeakPtr());
|
|
cookie_change_notifier_ = std::make_unique<CookieChangeNotifier>(this);
|
|
|
|
BrowserContextDependencyManager::GetInstance()->MarkBrowserContextLive(this);
|
|
}
|
|
|
|
AtomBrowserContext::~AtomBrowserContext() {
|
|
DCHECK_CURRENTLY_ON(BrowserThread::UI);
|
|
NotifyWillBeDestroyed(this);
|
|
ShutdownStoragePartitions();
|
|
io_handle_->ShutdownOnUIThread();
|
|
// Notify any keyed services of browser context destruction.
|
|
BrowserContextDependencyManager::GetInstance()->DestroyBrowserContextServices(
|
|
this);
|
|
}
|
|
|
|
void AtomBrowserContext::InitPrefs() {
|
|
auto prefs_path = GetPath().Append(FILE_PATH_LITERAL("Preferences"));
|
|
base::ThreadRestrictions::ScopedAllowIO allow_io;
|
|
PrefServiceFactory prefs_factory;
|
|
scoped_refptr<JsonPrefStore> pref_store =
|
|
base::MakeRefCounted<JsonPrefStore>(prefs_path);
|
|
pref_store->ReadPrefs(); // Synchronous.
|
|
prefs_factory.set_user_prefs(pref_store);
|
|
|
|
auto registry = WrapRefCounted(new PrefRegistrySimple);
|
|
|
|
registry->RegisterFilePathPref(prefs::kSelectFileLastDirectory,
|
|
base::FilePath());
|
|
base::FilePath download_dir;
|
|
base::PathService::Get(chrome::DIR_DEFAULT_DOWNLOADS, &download_dir);
|
|
registry->RegisterFilePathPref(prefs::kDownloadDefaultDirectory,
|
|
download_dir);
|
|
registry->RegisterDictionaryPref(prefs::kDevToolsFileSystemPaths);
|
|
InspectableWebContentsImpl::RegisterPrefs(registry.get());
|
|
MediaDeviceIDSalt::RegisterPrefs(registry.get());
|
|
ZoomLevelDelegate::RegisterPrefs(registry.get());
|
|
PrefProxyConfigTrackerImpl::RegisterPrefs(registry.get());
|
|
|
|
prefs_ = prefs_factory.Create(
|
|
registry.get(),
|
|
std::make_unique<PrefStoreDelegate>(weak_factory_.GetWeakPtr()));
|
|
prefs_->UpdateCommandLinePrefStore(new ValueMapPrefStore);
|
|
}
|
|
|
|
void AtomBrowserContext::SetUserAgent(const std::string& user_agent) {
|
|
user_agent_ = user_agent;
|
|
}
|
|
|
|
net::URLRequestContextGetter* AtomBrowserContext::CreateRequestContext(
|
|
content::ProtocolHandlerMap* protocol_handlers,
|
|
content::URLRequestInterceptorScopedVector protocol_interceptors) {
|
|
return io_handle_
|
|
->CreateMainRequestContextGetter(protocol_handlers,
|
|
std::move(protocol_interceptors))
|
|
.get();
|
|
}
|
|
|
|
net::URLRequestContextGetter* AtomBrowserContext::CreateMediaRequestContext() {
|
|
return io_handle_->GetMainRequestContextGetter().get();
|
|
}
|
|
|
|
net::URLRequestContextGetter* AtomBrowserContext::GetRequestContext() {
|
|
return GetDefaultStoragePartition(this)->GetURLRequestContext();
|
|
}
|
|
|
|
network::mojom::NetworkContextPtr AtomBrowserContext::GetNetworkContext() {
|
|
return io_handle_->GetNetworkContext();
|
|
}
|
|
|
|
base::FilePath AtomBrowserContext::GetPath() const {
|
|
return path_;
|
|
}
|
|
|
|
bool AtomBrowserContext::IsOffTheRecord() const {
|
|
return in_memory_;
|
|
}
|
|
|
|
bool AtomBrowserContext::CanUseHttpCache() const {
|
|
return use_cache_;
|
|
}
|
|
|
|
int AtomBrowserContext::GetMaxCacheSize() const {
|
|
return max_cache_size_;
|
|
}
|
|
|
|
content::ResourceContext* AtomBrowserContext::GetResourceContext() {
|
|
return io_handle_->GetResourceContext();
|
|
}
|
|
|
|
std::string AtomBrowserContext::GetMediaDeviceIDSalt() {
|
|
if (!media_device_id_salt_.get())
|
|
media_device_id_salt_.reset(new MediaDeviceIDSalt(prefs_.get()));
|
|
return media_device_id_salt_->GetSalt();
|
|
}
|
|
|
|
std::unique_ptr<content::ZoomLevelDelegate>
|
|
AtomBrowserContext::CreateZoomLevelDelegate(
|
|
const base::FilePath& partition_path) {
|
|
if (!IsOffTheRecord()) {
|
|
return std::make_unique<ZoomLevelDelegate>(prefs(), partition_path);
|
|
}
|
|
return std::unique_ptr<content::ZoomLevelDelegate>();
|
|
}
|
|
|
|
content::DownloadManagerDelegate*
|
|
AtomBrowserContext::GetDownloadManagerDelegate() {
|
|
if (!download_manager_delegate_.get()) {
|
|
auto* download_manager = content::BrowserContext::GetDownloadManager(this);
|
|
download_manager_delegate_.reset(
|
|
new AtomDownloadManagerDelegate(download_manager));
|
|
}
|
|
return download_manager_delegate_.get();
|
|
}
|
|
|
|
content::BrowserPluginGuestManager* AtomBrowserContext::GetGuestManager() {
|
|
if (!guest_manager_)
|
|
guest_manager_.reset(new WebViewManager);
|
|
return guest_manager_.get();
|
|
}
|
|
|
|
content::PermissionControllerDelegate*
|
|
AtomBrowserContext::GetPermissionControllerDelegate() {
|
|
if (!permission_manager_.get())
|
|
permission_manager_.reset(new AtomPermissionManager);
|
|
return permission_manager_.get();
|
|
}
|
|
|
|
storage::SpecialStoragePolicy* AtomBrowserContext::GetSpecialStoragePolicy() {
|
|
return storage_policy_.get();
|
|
}
|
|
|
|
std::string AtomBrowserContext::GetUserAgent() const {
|
|
return user_agent_;
|
|
}
|
|
|
|
AtomBlobReader* AtomBrowserContext::GetBlobReader() {
|
|
if (!blob_reader_.get()) {
|
|
content::ChromeBlobStorageContext* blob_context =
|
|
content::ChromeBlobStorageContext::GetFor(this);
|
|
blob_reader_.reset(new AtomBlobReader(blob_context));
|
|
}
|
|
return blob_reader_.get();
|
|
}
|
|
|
|
content::PushMessagingService* AtomBrowserContext::GetPushMessagingService() {
|
|
return nullptr;
|
|
}
|
|
|
|
content::SSLHostStateDelegate* AtomBrowserContext::GetSSLHostStateDelegate() {
|
|
return nullptr;
|
|
}
|
|
|
|
content::BackgroundFetchDelegate*
|
|
AtomBrowserContext::GetBackgroundFetchDelegate() {
|
|
return nullptr;
|
|
}
|
|
|
|
content::BackgroundSyncController*
|
|
AtomBrowserContext::GetBackgroundSyncController() {
|
|
return nullptr;
|
|
}
|
|
|
|
content::BrowsingDataRemoverDelegate*
|
|
AtomBrowserContext::GetBrowsingDataRemoverDelegate() {
|
|
return nullptr;
|
|
}
|
|
|
|
net::URLRequestContextGetter*
|
|
AtomBrowserContext::CreateRequestContextForStoragePartition(
|
|
const base::FilePath& partition_path,
|
|
bool in_memory,
|
|
content::ProtocolHandlerMap* protocol_handlers,
|
|
content::URLRequestInterceptorScopedVector request_interceptors) {
|
|
NOTREACHED();
|
|
return nullptr;
|
|
}
|
|
|
|
net::URLRequestContextGetter*
|
|
AtomBrowserContext::CreateMediaRequestContextForStoragePartition(
|
|
const base::FilePath& partition_path,
|
|
bool in_memory) {
|
|
NOTREACHED();
|
|
return nullptr;
|
|
}
|
|
|
|
ResolveProxyHelper* AtomBrowserContext::GetResolveProxyHelper() {
|
|
if (!resolve_proxy_helper_) {
|
|
resolve_proxy_helper_ = base::MakeRefCounted<ResolveProxyHelper>(this);
|
|
}
|
|
return resolve_proxy_helper_.get();
|
|
}
|
|
|
|
// static
|
|
scoped_refptr<AtomBrowserContext> AtomBrowserContext::From(
|
|
const std::string& partition,
|
|
bool in_memory,
|
|
const base::DictionaryValue& options) {
|
|
PartitionKey key(partition, in_memory);
|
|
auto* browser_context = browser_context_map_[key].get();
|
|
if (browser_context)
|
|
return scoped_refptr<AtomBrowserContext>(browser_context);
|
|
|
|
auto* new_context = new AtomBrowserContext(partition, in_memory, options);
|
|
browser_context_map_[key] = new_context->GetWeakPtr();
|
|
return scoped_refptr<AtomBrowserContext>(new_context);
|
|
}
|
|
|
|
} // namespace atom
|