// Copyright 2015 The Chromium Embedded Framework Authors. // Portions 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 "libcef/browser/extensions/extensions_browser_client.h" #include #include "libcef/browser/browser_context_impl.h" #include "libcef/browser/browser_host_impl.h" #include "libcef/browser/extensions/chrome_api_registration.h" #include "libcef/browser/extensions/component_extension_resource_manager.h" #include "libcef/browser/extensions/extension_system.h" #include "libcef/browser/extensions/extension_system_factory.h" #include "libcef/browser/extensions/extension_web_contents_observer.h" #include "libcef/browser/extensions/extensions_api_client.h" #include "libcef/browser/request_context_impl.h" //#include "cef/libcef/browser/extensions/api/generated_api_registration.h" #include "chrome/browser/browser_process.h" #include "chrome/browser/extensions/chrome_url_request_util.h" #include "chrome/browser/extensions/event_router_forwarder.h" #include "content/public/browser/browser_context.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/render_frame_host.h" #include "extensions/browser/api/extensions_api_client.h" #include "extensions/browser/api/generated_api_registration.h" #include "extensions/browser/api/runtime/runtime_api_delegate.h" #include "extensions/browser/app_sorting.h" #include "extensions/browser/event_router.h" #include "extensions/browser/extension_function_registry.h" #include "extensions/browser/extension_host_delegate.h" #include "extensions/browser/mojo/interface_registration.h" #include "extensions/browser/serial_extension_host_queue.h" #include "extensions/browser/url_request_util.h" #include "extensions/common/constants.h" using content::BrowserContext; using content::BrowserThread; namespace extensions { CefExtensionsBrowserClient::CefExtensionsBrowserClient() : api_client_(new CefExtensionsAPIClient), resource_manager_(new CefComponentExtensionResourceManager) {} CefExtensionsBrowserClient::~CefExtensionsBrowserClient() {} // static CefExtensionsBrowserClient* CefExtensionsBrowserClient::Get() { return static_cast( ExtensionsBrowserClient::Get()); } bool CefExtensionsBrowserClient::IsShuttingDown() { return false; } bool CefExtensionsBrowserClient::AreExtensionsDisabled( const base::CommandLine& command_line, BrowserContext* context) { return false; } bool CefExtensionsBrowserClient::IsValidContext(BrowserContext* context) { return CefBrowserContextImpl::GetForContext(context) != NULL; } bool CefExtensionsBrowserClient::IsSameContext(BrowserContext* first, BrowserContext* second) { // Returns true if |first| and |second| share the same underlying // CefBrowserContextImpl. return GetCefImplContext(first) == GetCefImplContext(second); } bool CefExtensionsBrowserClient::HasOffTheRecordContext( BrowserContext* context) { // CEF doesn't use incognito contexts. return false; } BrowserContext* CefExtensionsBrowserClient::GetOffTheRecordContext( BrowserContext* context) { return NULL; } BrowserContext* CefExtensionsBrowserClient::GetOriginalContext( BrowserContext* context) { return GetCefImplContext(context); } BrowserContext* CefExtensionsBrowserClient::GetCefImplContext( BrowserContext* context) { return CefBrowserContextImpl::GetForContext(context); } bool CefExtensionsBrowserClient::IsGuestSession(BrowserContext* context) const { return false; } bool CefExtensionsBrowserClient::IsExtensionIncognitoEnabled( const std::string& extension_id, content::BrowserContext* context) const { return false; } bool CefExtensionsBrowserClient::CanExtensionCrossIncognito( const Extension* extension, content::BrowserContext* context) const { return false; } net::URLRequestJob* CefExtensionsBrowserClient::MaybeCreateResourceBundleRequestJob( net::URLRequest* request, net::NetworkDelegate* network_delegate, const base::FilePath& directory_path, const std::string& content_security_policy, bool send_cors_header) { return chrome_url_request_util::MaybeCreateURLRequestResourceBundleJob( request, network_delegate, directory_path, content_security_policy, send_cors_header); } bool CefExtensionsBrowserClient::AllowCrossRendererResourceLoad( net::URLRequest* request, bool is_incognito, const Extension* extension, InfoMap* extension_info_map) { // TODO(cef): This bypasses additional checks added to // AllowCrossRendererResourceLoad() in https://crrev.com/5cf9d45c. Figure out // why permission is not being granted based on "web_accessible_resources" // specified in the PDF extension manifest.json file. if (extension && extension->id() == extension_misc::kPdfExtensionId) return true; bool allowed = false; if (url_request_util::AllowCrossRendererResourceLoad( request, is_incognito, extension, extension_info_map, &allowed)) { return allowed; } // Couldn't determine if resource is allowed. Block the load. return false; } PrefService* CefExtensionsBrowserClient::GetPrefServiceForContext( BrowserContext* context) { return static_cast(context)->GetPrefs(); } void CefExtensionsBrowserClient::GetEarlyExtensionPrefsObservers( content::BrowserContext* context, std::vector* observers) const {} ProcessManagerDelegate* CefExtensionsBrowserClient::GetProcessManagerDelegate() const { return NULL; } std::unique_ptr CefExtensionsBrowserClient::CreateExtensionHostDelegate() { // CEF does not use the ExtensionHost constructor that calls this method. NOTREACHED(); return std::unique_ptr(); } bool CefExtensionsBrowserClient::CreateBackgroundExtensionHost( const Extension* extension, content::BrowserContext* browser_context, const GURL& url, ExtensionHost** host) { // The BrowserContext referenced by ProcessManager should always be an *Impl. DCHECK(!static_cast(browser_context)->is_proxy()); CefBrowserContextImpl* browser_context_impl = CefBrowserContextImpl::GetForContext(browser_context); // A CEF representation should always exist. CefRefPtr cef_extension = browser_context_impl->extension_system()->GetExtension(extension->id()); DCHECK(cef_extension); if (!cef_extension) { // Cancel the background host creation. return true; } // Always use the same request context that the extension was registered with. // May represent an *Impl or *Proxy BrowserContext. // GetLoaderContext() will return NULL for internal extensions. CefRefPtr request_context = cef_extension->GetLoaderContext(); if (!request_context) { // Cancel the background host creation. return true; } CefBrowserHostImpl::CreateParams create_params; create_params.url = url; create_params.request_context = request_context; CefRefPtr handler = cef_extension->GetHandler(); if (handler.get() && handler->OnBeforeBackgroundBrowser( cef_extension, url.spec(), create_params.client, create_params.settings)) { // Cancel the background host creation. return true; } // This triggers creation of the background host. create_params.extension = extension; create_params.extension_host_type = extensions::VIEW_TYPE_EXTENSION_BACKGROUND_PAGE; // Browser creation may fail under certain rare circumstances. Fail the // background host creation in that case. CefRefPtr browser = CefBrowserHostImpl::Create(create_params); if (browser) { *host = browser->extension_host(); DCHECK(*host); } return true; } bool CefExtensionsBrowserClient::DidVersionUpdate(BrowserContext* context) { // TODO(jamescook): We might want to tell extensions when app_shell updates. return false; } void CefExtensionsBrowserClient::PermitExternalProtocolHandler() {} bool CefExtensionsBrowserClient::IsRunningInForcedAppMode() { return false; } bool CefExtensionsBrowserClient::IsLoggedInAsPublicAccount() { return false; } ExtensionSystemProvider* CefExtensionsBrowserClient::GetExtensionSystemFactory() { return CefExtensionSystemFactory::GetInstance(); } void CefExtensionsBrowserClient::RegisterExtensionFunctions( ExtensionFunctionRegistry* registry) const { // Register core extension-system APIs. api::GeneratedFunctionRegistry::RegisterAll(registry); // CEF-only APIs. // TODO(cef): Enable if/when CEF exposes its own Mojo APIs. See // libcef/common/extensions/api/README.txt for details. // api::cef::CefGeneratedFunctionRegistry::RegisterAll(registry); // Chrome APIs whitelisted by CEF. api::cef::ChromeFunctionRegistry::RegisterAll(registry); } void CefExtensionsBrowserClient::RegisterExtensionInterfaces( service_manager::BinderRegistryWithArgs* registry, content::RenderFrameHost* render_frame_host, const Extension* extension) const { RegisterInterfacesForExtension(registry, render_frame_host, extension); } std::unique_ptr CefExtensionsBrowserClient::CreateRuntimeAPIDelegate( content::BrowserContext* context) const { // TODO(extensions): Implement to support Apps. NOTREACHED(); return nullptr; } const ComponentExtensionResourceManager* CefExtensionsBrowserClient::GetComponentExtensionResourceManager() { return resource_manager_.get(); } void CefExtensionsBrowserClient::BroadcastEventToRenderers( events::HistogramValue histogram_value, const std::string& event_name, std::unique_ptr args) { g_browser_process->extension_event_router_forwarder() ->BroadcastEventToRenderers(histogram_value, event_name, std::move(args), GURL()); } net::NetLog* CefExtensionsBrowserClient::GetNetLog() { return NULL; } ExtensionCache* CefExtensionsBrowserClient::GetExtensionCache() { // Only used by Chrome via ExtensionService. NOTREACHED(); return NULL; } bool CefExtensionsBrowserClient::IsBackgroundUpdateAllowed() { return true; } bool CefExtensionsBrowserClient::IsMinBrowserVersionSupported( const std::string& min_version) { return true; } ExtensionWebContentsObserver* CefExtensionsBrowserClient::GetExtensionWebContentsObserver( content::WebContents* web_contents) { return CefExtensionWebContentsObserver::FromWebContents(web_contents); } KioskDelegate* CefExtensionsBrowserClient::GetKioskDelegate() { NOTREACHED(); return NULL; } bool CefExtensionsBrowserClient::IsLockScreenContext( content::BrowserContext* context) { return false; } std::string CefExtensionsBrowserClient::GetApplicationLocale() { return g_browser_process->GetApplicationLocale(); } ExtensionHostQueue* CefExtensionsBrowserClient::GetExtensionHostQueue() { if (!extension_host_queue_) extension_host_queue_.reset(new SerialExtensionHostQueue); return extension_host_queue_.get(); } } // namespace extensions