blob: 3b247a96475a9b9d6e5be1ea6210ebb2dee6783e [file] [log] [blame]
// 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 "android_webview/browser/aw_browser_context.h"
#include <memory>
#include <string>
#include <utility>
#include "android_webview/browser/aw_download_manager_delegate.h"
#include "android_webview/browser/aw_form_database_service.h"
#include "android_webview/browser/aw_metrics_service_client.h"
#include "android_webview/browser/aw_permission_manager.h"
#include "android_webview/browser/aw_quota_manager_bridge.h"
#include "android_webview/browser/aw_resource_context.h"
#include "android_webview/browser/aw_safe_browsing_whitelist_manager.h"
#include "android_webview/browser/aw_web_ui_controller_factory.h"
#include "android_webview/browser/net/aw_url_request_context_getter.h"
#include "base/base_paths_posix.h"
#include "base/bind.h"
#include "base/path_service.h"
#include "base/single_thread_task_runner.h"
#include "base/task/post_task.h"
#include "components/policy/core/browser/browser_policy_connector_base.h"
#include "components/prefs/pref_service.h"
#include "components/safe_browsing/triggers/trigger_manager.h"
#include "components/url_formatter/url_fixer.h"
#include "components/user_prefs/user_prefs.h"
#include "components/visitedlink/browser/visitedlink_master.h"
#include "content/public/browser/browser_task_traits.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/ssl_host_state_delegate.h"
#include "content/public/browser/storage_partition.h"
#include "content/public/browser/web_contents.h"
#include "net/proxy_resolution/proxy_config_service_android.h"
#include "net/proxy_resolution/proxy_resolution_service.h"
#include "services/preferences/tracked/segregated_pref_store.h"
using base::FilePath;
using content::BrowserThread;
namespace android_webview {
namespace prefs {
// String that specifies the Android account type to use for Negotiate
// authentication.
const char kAuthAndroidNegotiateAccountType[] =
"auth.android_negotiate_account_type";
// Whitelist containing servers for which Integrated Authentication is enabled.
const char kAuthServerWhitelist[] = "auth.server_whitelist";
const char kWebRestrictionsAuthority[] = "web_restrictions_authority";
} // namespace prefs
namespace {
const base::FilePath::CharType kChannelIDFilename[] = "Origin Bound Certs";
const void* const kDownloadManagerDelegateKey = &kDownloadManagerDelegateKey;
AwBrowserContext* g_browser_context = NULL;
std::unique_ptr<net::ProxyConfigServiceAndroid> CreateProxyConfigService() {
std::unique_ptr<net::ProxyConfigServiceAndroid> config_service_android =
std::make_unique<net::ProxyConfigServiceAndroid>(
base::CreateSingleThreadTaskRunnerWithTraits({BrowserThread::IO}),
base::ThreadTaskRunnerHandle::Get());
// TODO(csharrison) Architect the wrapper better so we don't need a cast for
// android ProxyConfigServices.
config_service_android->set_exclude_pac_url(true);
return config_service_android;
}
std::unique_ptr<AwSafeBrowsingWhitelistManager>
CreateSafeBrowsingWhitelistManager() {
// Should not be called until the end of PreMainMessageLoopRun,
scoped_refptr<base::SequencedTaskRunner> background_task_runner =
base::CreateSequencedTaskRunnerWithTraits(
{base::MayBlock(), base::TaskPriority::BEST_EFFORT});
scoped_refptr<base::SingleThreadTaskRunner> io_task_runner =
base::CreateSingleThreadTaskRunnerWithTraits({BrowserThread::IO});
return std::make_unique<AwSafeBrowsingWhitelistManager>(
background_task_runner, io_task_runner);
}
} // namespace
AwBrowserContext::AwBrowserContext(
const base::FilePath path,
std::unique_ptr<PrefService> pref_service,
std::unique_ptr<policy::BrowserPolicyConnectorBase> policy_connector)
: context_storage_path_(path),
user_pref_service_(std::move(pref_service)),
browser_policy_connector_(std::move(policy_connector)) {
DCHECK(!g_browser_context);
g_browser_context = this;
BrowserContext::Initialize(this, path);
pref_change_registrar_.Init(user_pref_service_.get());
user_prefs::UserPrefs::Set(this, user_pref_service_.get());
// This constructor is entered during the creation of ContentBrowserClient,
// before browser threads are created. Therefore any checks to enforce
// threading (such as BrowserThread::CurrentlyOn()) will fail here.
}
AwBrowserContext::~AwBrowserContext() {
DCHECK_EQ(this, g_browser_context);
g_browser_context = NULL;
}
// static
AwBrowserContext* AwBrowserContext::GetDefault() {
// TODO(joth): rather than store in a global here, lookup this instance
// from the Java-side peer.
return g_browser_context;
}
// static
AwBrowserContext* AwBrowserContext::FromWebContents(
content::WebContents* web_contents) {
// This is safe; this is the only implementation of the browser context.
return static_cast<AwBrowserContext*>(web_contents->GetBrowserContext());
}
// static
base::FilePath AwBrowserContext::GetCacheDir() {
FilePath cache_path;
base::PathService::Get(base::DIR_CACHE, &cache_path);
cache_path =
cache_path.Append(FILE_PATH_LITERAL("org.chromium.android_webview"));
return cache_path;
}
void AwBrowserContext::PreMainMessageLoopRun(net::NetLog* net_log) {
FilePath cache_path = GetCacheDir();
// TODO(ntfschr): set this to nullptr when the NetworkService is disabled,
// once we remove a dependency on url_request_context_getter_
// (http://crbug.com/887538).
url_request_context_getter_ = new AwURLRequestContextGetter(
cache_path, context_storage_path_.Append(kChannelIDFilename),
CreateProxyConfigService(), user_pref_service_.get(), net_log);
scoped_refptr<base::SequencedTaskRunner> db_task_runner =
base::CreateSequencedTaskRunnerWithTraits(
{base::MayBlock(), base::TaskPriority::BEST_EFFORT,
base::TaskShutdownBehavior::SKIP_ON_SHUTDOWN});
visitedlink_master_.reset(
new visitedlink::VisitedLinkMaster(this, this, false));
visitedlink_master_->Init();
form_database_service_.reset(
new AwFormDatabaseService(context_storage_path_));
EnsureResourceContextInitialized(this);
web_restriction_provider_.reset(
new web_restrictions::WebRestrictionsClient());
pref_change_registrar_.Add(
prefs::kWebRestrictionsAuthority,
base::BindRepeating(&AwBrowserContext::OnWebRestrictionsAuthorityChanged,
base::Unretained(this)));
web_restriction_provider_->SetAuthority(
user_pref_service_->GetString(prefs::kWebRestrictionsAuthority));
safe_browsing_ui_manager_ = new AwSafeBrowsingUIManager(
GetAwURLRequestContext(), user_pref_service_.get());
safe_browsing_db_manager_ =
new safe_browsing::RemoteSafeBrowsingDatabaseManager();
safe_browsing_trigger_manager_ =
std::make_unique<safe_browsing::TriggerManager>(
safe_browsing_ui_manager_.get(),
/*referrer_chain_provider=*/nullptr,
/*local_state_prefs=*/nullptr);
safe_browsing_whitelist_manager_ = CreateSafeBrowsingWhitelistManager();
content::WebUIControllerFactory::RegisterFactory(
AwWebUIControllerFactory::GetInstance());
}
void AwBrowserContext::OnWebRestrictionsAuthorityChanged() {
web_restriction_provider_->SetAuthority(
user_pref_service_->GetString(prefs::kWebRestrictionsAuthority));
}
void AwBrowserContext::AddVisitedURLs(const std::vector<GURL>& urls) {
DCHECK(visitedlink_master_);
visitedlink_master_->AddURLs(urls);
}
AwQuotaManagerBridge* AwBrowserContext::GetQuotaManagerBridge() {
if (!quota_manager_bridge_.get()) {
quota_manager_bridge_ = AwQuotaManagerBridge::Create(this);
}
return quota_manager_bridge_.get();
}
AwFormDatabaseService* AwBrowserContext::GetFormDatabaseService() {
return form_database_service_.get();
}
AwURLRequestContextGetter* AwBrowserContext::GetAwURLRequestContext() {
return url_request_context_getter_.get();
}
base::FilePath AwBrowserContext::GetPath() const {
return context_storage_path_;
}
bool AwBrowserContext::IsOffTheRecord() const {
// Android WebView does not support off the record profile yet.
return false;
}
content::ResourceContext* AwBrowserContext::GetResourceContext() {
if (!resource_context_) {
resource_context_.reset(new AwResourceContext);
}
return resource_context_.get();
}
content::DownloadManagerDelegate*
AwBrowserContext::GetDownloadManagerDelegate() {
if (!GetUserData(kDownloadManagerDelegateKey)) {
SetUserData(kDownloadManagerDelegateKey,
std::make_unique<AwDownloadManagerDelegate>());
}
return static_cast<AwDownloadManagerDelegate*>(
GetUserData(kDownloadManagerDelegateKey));
}
content::BrowserPluginGuestManager* AwBrowserContext::GetGuestManager() {
return NULL;
}
storage::SpecialStoragePolicy* AwBrowserContext::GetSpecialStoragePolicy() {
// Intentionally returning NULL as 'Extensions' and 'Apps' not supported.
return NULL;
}
content::PushMessagingService* AwBrowserContext::GetPushMessagingService() {
// TODO(johnme): Support push messaging in WebView.
return NULL;
}
content::SSLHostStateDelegate* AwBrowserContext::GetSSLHostStateDelegate() {
if (!ssl_host_state_delegate_.get()) {
ssl_host_state_delegate_.reset(new AwSSLHostStateDelegate());
}
return ssl_host_state_delegate_.get();
}
content::PermissionControllerDelegate*
AwBrowserContext::GetPermissionControllerDelegate() {
if (!permission_manager_.get())
permission_manager_.reset(new AwPermissionManager());
return permission_manager_.get();
}
content::BackgroundFetchDelegate*
AwBrowserContext::GetBackgroundFetchDelegate() {
// TODO(crbug.com/766077): Resolve whether to support or disable background
// fetch on WebView.
return nullptr;
}
content::BackgroundSyncController*
AwBrowserContext::GetBackgroundSyncController() {
return nullptr;
}
content::BrowsingDataRemoverDelegate*
AwBrowserContext::GetBrowsingDataRemoverDelegate() {
return nullptr;
}
net::URLRequestContextGetter* AwBrowserContext::CreateRequestContext(
content::ProtocolHandlerMap* protocol_handlers,
content::URLRequestInterceptorScopedVector request_interceptors) {
// This function cannot actually create the request context because
// there is a reentrant dependency on GetResourceContext() via
// content::StoragePartitionImplMap::Create(). This is not fixable
// until http://crbug.com/159193. Until then, assert that the context
// has already been allocated and just handle setting the protocol_handlers.
DCHECK(url_request_context_getter_.get());
url_request_context_getter_->SetHandlersAndInterceptors(
protocol_handlers, std::move(request_interceptors));
return url_request_context_getter_.get();
}
net::URLRequestContextGetter*
AwBrowserContext::CreateRequestContextForStoragePartition(
const base::FilePath& partition_path,
bool in_memory,
content::ProtocolHandlerMap* protocol_handlers,
content::URLRequestInterceptorScopedVector request_interceptors) {
NOTREACHED();
return NULL;
}
net::URLRequestContextGetter* AwBrowserContext::CreateMediaRequestContext() {
return url_request_context_getter_.get();
}
net::URLRequestContextGetter*
AwBrowserContext::CreateMediaRequestContextForStoragePartition(
const base::FilePath& partition_path,
bool in_memory) {
NOTREACHED();
return NULL;
}
web_restrictions::WebRestrictionsClient*
AwBrowserContext::GetWebRestrictionProvider() {
DCHECK(web_restriction_provider_);
return web_restriction_provider_.get();
}
AwSafeBrowsingUIManager* AwBrowserContext::GetSafeBrowsingUIManager() const {
return safe_browsing_ui_manager_.get();
}
safe_browsing::RemoteSafeBrowsingDatabaseManager*
AwBrowserContext::GetSafeBrowsingDBManager() {
DCHECK_CURRENTLY_ON(BrowserThread::IO);
if (!safe_browsing_db_manager_started_) {
// V4ProtocolConfig is not used. Just create one with empty values..
safe_browsing::V4ProtocolConfig config("", false, "", "");
safe_browsing_db_manager_->StartOnIOThread(
safe_browsing_ui_manager_->GetURLLoaderFactoryOnIOThread(), config);
safe_browsing_db_manager_started_ = true;
}
return safe_browsing_db_manager_.get();
}
safe_browsing::TriggerManager* AwBrowserContext::GetSafeBrowsingTriggerManager()
const {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
return safe_browsing_trigger_manager_.get();
}
AwSafeBrowsingWhitelistManager*
AwBrowserContext::GetSafeBrowsingWhitelistManager() const {
// Should not be called until the end of PreMainMessageLoopRun,
return safe_browsing_whitelist_manager_.get();
}
void AwBrowserContext::RebuildTable(
const scoped_refptr<URLEnumerator>& enumerator) {
// Android WebView rebuilds from WebChromeClient.getVisitedHistory. The client
// can change in the lifetime of this WebView and may not yet be set here.
// Therefore this initialization path is not used.
enumerator->OnComplete(true);
}
} // namespace android_webview