blob: 36f8651c95f2e18d983953f3e8735b2520473057 [file] [log] [blame]
// Copyright (c) 2009 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/extensions/extension_function_dispatcher.h"
#include "base/process_util.h"
#include "base/singleton.h"
#include "base/values.h"
#include "chrome/browser/browser.h"
#include "chrome/browser/browser_window.h"
#include "chrome/browser/extensions/execute_code_in_tab_function.h"
#include "chrome/browser/extensions/extension_accessibility_api.h"
#include "chrome/browser/extensions/extension_bookmark_manager_api.h"
#include "chrome/browser/extensions/extension_bookmarks_module.h"
#include "chrome/browser/extensions/extension_bookmarks_module_constants.h"
#include "chrome/browser/extensions/extension_browser_actions_api.h"
#include "chrome/browser/extensions/extension_dom_ui.h"
#include "chrome/browser/extensions/extension_function.h"
#include "chrome/browser/extensions/extension_history_api.h"
#include "chrome/browser/extensions/extension_i18n_api.h"
#include "chrome/browser/extensions/extension_message_service.h"
#include "chrome/browser/extensions/extension_metrics_module.h"
#include "chrome/browser/extensions/extension_page_actions_module.h"
#include "chrome/browser/extensions/extension_page_actions_module_constants.h"
#include "chrome/browser/extensions/extension_popup_api.h"
#include "chrome/browser/extensions/extension_process_manager.h"
#include "chrome/browser/extensions/extension_processes_api.h"
#include "chrome/browser/extensions/extension_tabs_module.h"
#include "chrome/browser/extensions/extension_tabs_module_constants.h"
#include "chrome/browser/extensions/extension_test_api.h"
#include "chrome/browser/extensions/extension_toolstrip_api.h"
#include "chrome/browser/extensions/extensions_quota_service.h"
#include "chrome/browser/extensions/extensions_service.h"
#include "chrome/browser/profile.h"
#include "chrome/browser/renderer_host/render_process_host.h"
#include "chrome/browser/renderer_host/render_view_host.h"
#include "chrome/common/chrome_switches.h"
#include "chrome/common/render_messages.h"
#include "chrome/common/result_codes.h"
#include "chrome/common/url_constants.h"
// FactoryRegistry -------------------------------------------------------------
namespace {
// Template for defining ExtensionFunctionFactory.
template<class T>
ExtensionFunction* NewExtensionFunction() {
return new T();
}
// Contains a list of all known extension functions and allows clients to
// create instances of them.
class FactoryRegistry {
public:
static FactoryRegistry* instance();
FactoryRegistry() { ResetFunctions(); }
// Resets all functions to their default values.
void ResetFunctions();
// Adds all function names to 'names'.
void GetAllNames(std::vector<std::string>* names);
// Allows overriding of specific functions (e.g. for testing). Functions
// must be previously registered. Returns true if successful.
bool OverrideFunction(const std::string& name,
ExtensionFunctionFactory factory);
// Factory method for the ExtensionFunction registered as 'name'.
ExtensionFunction* NewFunction(const std::string& name);
private:
template<class T>
void RegisterFunction() {
factories_[T::function_name()] = &NewExtensionFunction<T>;
}
typedef std::map<std::string, ExtensionFunctionFactory> FactoryMap;
FactoryMap factories_;
};
FactoryRegistry* FactoryRegistry::instance() {
return Singleton<FactoryRegistry>::get();
}
void FactoryRegistry::ResetFunctions() {
// Register all functions here.
// Windows
RegisterFunction<GetWindowFunction>();
RegisterFunction<GetCurrentWindowFunction>();
RegisterFunction<GetLastFocusedWindowFunction>();
RegisterFunction<GetAllWindowsFunction>();
RegisterFunction<CreateWindowFunction>();
RegisterFunction<UpdateWindowFunction>();
RegisterFunction<RemoveWindowFunction>();
// Tabs
RegisterFunction<GetTabFunction>();
RegisterFunction<GetSelectedTabFunction>();
RegisterFunction<GetAllTabsInWindowFunction>();
RegisterFunction<CreateTabFunction>();
RegisterFunction<UpdateTabFunction>();
RegisterFunction<MoveTabFunction>();
RegisterFunction<RemoveTabFunction>();
RegisterFunction<DetectTabLanguageFunction>();
RegisterFunction<CaptureVisibleTabFunction>();
RegisterFunction<TabsExecuteScriptFunction>();
RegisterFunction<TabsInsertCSSFunction>();
// Page Actions.
RegisterFunction<EnablePageActionFunction>();
RegisterFunction<DisablePageActionFunction>();
RegisterFunction<PageActionShowFunction>();
RegisterFunction<PageActionHideFunction>();
RegisterFunction<PageActionSetIconFunction>();
RegisterFunction<PageActionSetTitleFunction>();
RegisterFunction<PageActionSetPopupFunction>();
// Browser Actions.
RegisterFunction<BrowserActionSetIconFunction>();
RegisterFunction<BrowserActionSetTitleFunction>();
RegisterFunction<BrowserActionSetBadgeTextFunction>();
RegisterFunction<BrowserActionSetBadgeBackgroundColorFunction>();
RegisterFunction<BrowserActionSetPopupFunction>();
// Bookmarks.
RegisterFunction<GetBookmarksFunction>();
RegisterFunction<GetBookmarkChildrenFunction>();
RegisterFunction<GetBookmarkRecentFunction>();
RegisterFunction<GetBookmarkTreeFunction>();
RegisterFunction<SearchBookmarksFunction>();
RegisterFunction<RemoveBookmarkFunction>();
RegisterFunction<RemoveTreeBookmarkFunction>();
RegisterFunction<CreateBookmarkFunction>();
RegisterFunction<MoveBookmarkFunction>();
RegisterFunction<UpdateBookmarkFunction>();
// BookmarkManager
RegisterFunction<CopyBookmarkManagerFunction>();
RegisterFunction<CutBookmarkManagerFunction>();
RegisterFunction<PasteBookmarkManagerFunction>();
RegisterFunction<CanPasteBookmarkManagerFunction>();
RegisterFunction<ImportBookmarksFunction>();
RegisterFunction<ExportBookmarksFunction>();
RegisterFunction<SortChildrenBookmarkManagerFunction>();
RegisterFunction<BookmarkManagerGetStringsFunction>();
RegisterFunction<StartDragBookmarkManagerFunction>();
RegisterFunction<DropBookmarkManagerFunction>();
// History
RegisterFunction<AddUrlHistoryFunction>();
RegisterFunction<DeleteAllHistoryFunction>();
RegisterFunction<DeleteRangeHistoryFunction>();
RegisterFunction<DeleteUrlHistoryFunction>();
RegisterFunction<GetVisitsHistoryFunction>();
RegisterFunction<SearchHistoryFunction>();
// Toolstrips.
RegisterFunction<ToolstripExpandFunction>();
RegisterFunction<ToolstripCollapseFunction>();
// I18N.
RegisterFunction<GetAcceptLanguagesFunction>();
// Popup API.
RegisterFunction<PopupShowFunction>();
// Processes.
RegisterFunction<GetProcessForTabFunction>();
// Metrics.
if (CommandLine::ForCurrentProcess()->HasSwitch(
switches::kEnableMetricsExtensionApi)) {
RegisterFunction<MetricsRecordUserActionFunction>();
RegisterFunction<MetricsRecordValueFunction>();
RegisterFunction<MetricsRecordPercentageFunction>();
RegisterFunction<MetricsRecordCountFunction>();
RegisterFunction<MetricsRecordSmallCountFunction>();
RegisterFunction<MetricsRecordMediumCountFunction>();
RegisterFunction<MetricsRecordTimeFunction>();
RegisterFunction<MetricsRecordMediumTimeFunction>();
RegisterFunction<MetricsRecordLongTimeFunction>();
}
// Test.
RegisterFunction<ExtensionTestPassFunction>();
RegisterFunction<ExtensionTestFailFunction>();
RegisterFunction<ExtensionTestLogFunction>();
RegisterFunction<ExtensionTestQuotaResetFunction>();
RegisterFunction<ExtensionTestCreateIncognitoTabFunction>();
// Accessibility.
RegisterFunction<GetFocusedControlFunction>();
RegisterFunction<SetAccessibilityEnabledFunction>();
}
void FactoryRegistry::GetAllNames(std::vector<std::string>* names) {
for (FactoryMap::iterator iter = factories_.begin();
iter != factories_.end(); ++iter) {
names->push_back(iter->first);
}
}
bool FactoryRegistry::OverrideFunction(const std::string& name,
ExtensionFunctionFactory factory) {
FactoryMap::iterator iter = factories_.find(name);
if (iter == factories_.end()) {
return false;
} else {
iter->second = factory;
return true;
}
}
ExtensionFunction* FactoryRegistry::NewFunction(const std::string& name) {
FactoryMap::iterator iter = factories_.find(name);
DCHECK(iter != factories_.end());
ExtensionFunction* function = iter->second();
function->set_name(name);
return function;
}
}; // namespace
// ExtensionFunctionDispatcher::Delegate ---------------------------------------
gfx::NativeWindow ExtensionFunctionDispatcher::Delegate::
GetFrameNativeWindow() {
Browser* browser = GetBrowser(true);
// If a browser is bound to this dispatcher, then return the widget hosting
// the window. Extensions hosted in ExternalTabContainer objects may not
// have a running browser instance.
if (browser)
return browser->window()->GetNativeHandle();
return NULL;
}
// ExtensionFunctionDispatcher -------------------------------------------------
void ExtensionFunctionDispatcher::GetAllFunctionNames(
std::vector<std::string>* names) {
FactoryRegistry::instance()->GetAllNames(names);
}
bool ExtensionFunctionDispatcher::OverrideFunction(
const std::string& name, ExtensionFunctionFactory factory) {
return FactoryRegistry::instance()->OverrideFunction(name, factory);
}
void ExtensionFunctionDispatcher::ResetFunctions() {
FactoryRegistry::instance()->ResetFunctions();
}
std::set<ExtensionFunctionDispatcher*>*
ExtensionFunctionDispatcher::all_instances() {
static std::set<ExtensionFunctionDispatcher*> instances;
return &instances;
}
ExtensionFunctionDispatcher::ExtensionFunctionDispatcher(
RenderViewHost* render_view_host,
Delegate* delegate,
const GURL& url)
: profile_(render_view_host->process()->profile()),
render_view_host_(render_view_host),
delegate_(delegate),
url_(url),
ALLOW_THIS_IN_INITIALIZER_LIST(peer_(new Peer(this))) {
// TODO(erikkay) should we do something for these errors in Release?
DCHECK(url.SchemeIs(chrome::kExtensionScheme));
Extension* extension =
profile()->GetExtensionsService()->GetExtensionByURL(url);
DCHECK(extension);
all_instances()->insert(this);
// Notify the ExtensionProcessManager that the view was created.
ExtensionProcessManager* epm = profile()->GetExtensionProcessManager();
epm->RegisterExtensionProcess(extension_id(),
render_view_host->process()->id());
bool incognito_enabled =
profile()->GetExtensionsService()->IsIncognitoEnabled(extension->id());
// Update the extension permissions. Doing this each time we create an EFD
// ensures that new processes are informed of permissions for newly installed
// extensions.
render_view_host->Send(new ViewMsg_Extension_SetAPIPermissions(
extension->id(), extension->api_permissions()));
render_view_host->Send(new ViewMsg_Extension_SetHostPermissions(
extension->url(), extension->host_permissions()));
render_view_host->Send(new ViewMsg_Extension_ExtensionSetIncognitoEnabled(
extension->id(), incognito_enabled));
NotificationService::current()->Notify(
NotificationType::EXTENSION_FUNCTION_DISPATCHER_CREATED,
Source<Profile>(profile_),
Details<ExtensionFunctionDispatcher>(this));
}
ExtensionFunctionDispatcher::~ExtensionFunctionDispatcher() {
all_instances()->erase(this);
peer_->dispatcher_ = NULL;
NotificationService::current()->Notify(
NotificationType::EXTENSION_FUNCTION_DISPATCHER_DESTROYED,
Source<Profile>(profile_),
Details<ExtensionFunctionDispatcher>(this));
}
Browser* ExtensionFunctionDispatcher::GetBrowser(bool include_incognito) {
return delegate_->GetBrowser(include_incognito);
}
ExtensionPopupHost* ExtensionFunctionDispatcher::GetPopupHost() {
ExtensionHost* extension_host = GetExtensionHost();
if (extension_host) {
DCHECK(!GetExtensionDOMUI()) <<
"Function dispatcher registered in too many environments.";
return extension_host->popup_host();
} else {
ExtensionDOMUI* dom_ui = GetExtensionDOMUI();
return dom_ui->popup_host();
}
}
ExtensionHost* ExtensionFunctionDispatcher::GetExtensionHost() {
return delegate_->GetExtensionHost();
}
ExtensionDOMUI* ExtensionFunctionDispatcher::GetExtensionDOMUI() {
return delegate_->GetExtensionDOMUI();
}
Extension* ExtensionFunctionDispatcher::GetExtension() {
ExtensionsService* service = profile()->GetExtensionsService();
DCHECK(service);
Extension* extension = service->GetExtensionById(extension_id(), false);
DCHECK(extension);
return extension;
}
void ExtensionFunctionDispatcher::HandleRequest(const std::string& name,
const Value* args,
int request_id,
bool has_callback) {
scoped_refptr<ExtensionFunction> function(
FactoryRegistry::instance()->NewFunction(name));
function->set_dispatcher_peer(peer_);
function->SetArgs(args);
function->set_request_id(request_id);
function->set_has_callback(has_callback);
function->set_include_incognito(
profile()->GetExtensionsService()->IsIncognitoEnabled(extension_id()));
ExtensionsService* service = profile()->GetExtensionsService();
DCHECK(service);
ExtensionsQuotaService* quota = service->quota_service();
if (quota->Assess(extension_id(), function, args, base::TimeTicks::Now())) {
function->Run();
} else {
render_view_host_->SendExtensionResponse(function->request_id(), false,
std::string(), QuotaLimitHeuristic::kGenericOverQuotaError);
}
}
void ExtensionFunctionDispatcher::SendResponse(ExtensionFunction* function,
bool success) {
render_view_host_->SendExtensionResponse(function->request_id(), success,
function->GetResult(), function->GetError());
}
void ExtensionFunctionDispatcher::HandleBadMessage(ExtensionFunction* api) {
LOG(ERROR) << "bad extension message " <<
api->name() <<
" : terminating renderer.";
if (RenderProcessHost::run_renderer_in_process()) {
// In single process mode it is better if we don't suicide but just crash.
CHECK(false);
} else {
NOTREACHED();
base::KillProcess(render_view_host_->process()->GetHandle(),
ResultCodes::KILLED_BAD_MESSAGE, false);
}
}
Profile* ExtensionFunctionDispatcher::profile() {
return profile_;
}
gfx::NativeWindow ExtensionFunctionDispatcher::GetFrameNativeWindow() {
return delegate_ ? delegate_->GetFrameNativeWindow() : NULL;
}