// 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/renderer/extensions/extension_helper.h" #include "base/bind.h" #include "base/bind_helpers.h" #include "base/command_line.h" #include "base/json/json_string_value_serializer.h" #include "base/lazy_instance.h" #include "base/message_loop.h" #include "base/utf_string_conversions.h" #include "chrome/common/chrome_switches.h" #include "chrome/common/extensions/extension_messages.h" #include "chrome/common/render_messages.h" #include "chrome/common/url_constants.h" #include "chrome/renderer/extensions/chrome_v8_context.h" #include "chrome/renderer/extensions/console.h" #include "chrome/renderer/extensions/dispatcher.h" #include "chrome/renderer/extensions/miscellaneous_bindings.h" #include "chrome/renderer/extensions/user_script_scheduler.h" #include "chrome/renderer/extensions/user_script_slave.h" #include "content/public/renderer/render_view.h" #include "content/public/renderer/render_view_visitor.h" #include "extensions/common/constants.h" #include "third_party/WebKit/Source/Platform/chromium/public/WebURLRequest.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebConsoleMessage.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebDocument.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebScopedUserGesture.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebView.h" #include "webkit/glue/image_resource_fetcher.h" #include "webkit/glue/resource_fetcher.h" namespace base { class ListValue; } // namespace base using content::ConsoleMessageLevel; using WebKit::WebConsoleMessage; using WebKit::WebDataSource; using WebKit::WebFrame; using WebKit::WebURLRequest; using WebKit::WebScopedUserGesture; using WebKit::WebView; using webkit_glue::ImageResourceFetcher; using webkit_glue::ResourceFetcher; namespace extensions { namespace { // Keeps a mapping from the frame pointer to a UserScriptScheduler object. // We store this mapping per process, because a frame can jump from one // document to another with adoptNode, and so having the object be a // RenderViewObserver means it might miss some notifications after it moves. typedef std::map SchedulerMap; static base::LazyInstance g_schedulers = LAZY_INSTANCE_INITIALIZER; // A RenderViewVisitor class that iterates through the set of available // views, looking for a view of the given type, in the given browser window // and within the given extension. // Used to accumulate the list of views associated with an extension. class ViewAccumulator : public content::RenderViewVisitor { public: ViewAccumulator(const std::string& extension_id, int browser_window_id, ViewType view_type) : extension_id_(extension_id), browser_window_id_(browser_window_id), view_type_(view_type) { } std::vector views() { return views_; } // Returns false to terminate the iteration. virtual bool Visit(content::RenderView* render_view) OVERRIDE { ExtensionHelper* helper = ExtensionHelper::Get(render_view); if (!ViewTypeMatches(helper->view_type(), view_type_)) return true; GURL url = render_view->GetWebView()->mainFrame()->document().url(); if (!url.SchemeIs(kExtensionScheme)) return true; const std::string& extension_id = url.host(); if (extension_id != extension_id_) return true; if (browser_window_id_ != extension_misc::kUnknownWindowId && helper->browser_window_id() != browser_window_id_) { return true; } views_.push_back(render_view); if (view_type_ == VIEW_TYPE_EXTENSION_BACKGROUND_PAGE) return false; // There can be only one... return true; } private: // Returns true if |type| "isa" |match|. static bool ViewTypeMatches(ViewType type, ViewType match) { if (type == match) return true; // INVALID means match all. if (match == VIEW_TYPE_INVALID) return true; return false; } std::string extension_id_; int browser_window_id_; ViewType view_type_; std::vector views_; }; } // namespace // static std::vector ExtensionHelper::GetExtensionViews( const std::string& extension_id, int browser_window_id, ViewType view_type) { ViewAccumulator accumulator(extension_id, browser_window_id, view_type); content::RenderView::ForEach(&accumulator); return accumulator.views(); } // static content::RenderView* ExtensionHelper::GetBackgroundPage( const std::string& extension_id) { ViewAccumulator accumulator(extension_id, extension_misc::kUnknownWindowId, VIEW_TYPE_EXTENSION_BACKGROUND_PAGE); content::RenderView::ForEach(&accumulator); CHECK_LE(accumulator.views().size(), 1u); if (accumulator.views().size() == 0) return NULL; return accumulator.views()[0]; } ExtensionHelper::ExtensionHelper(content::RenderView* render_view, Dispatcher* dispatcher) : content::RenderViewObserver(render_view), content::RenderViewObserverTracker(render_view), dispatcher_(dispatcher), pending_app_icon_requests_(0), view_type_(VIEW_TYPE_INVALID), tab_id_(-1), browser_window_id_(-1) { } ExtensionHelper::~ExtensionHelper() { } bool ExtensionHelper::OnMessageReceived(const IPC::Message& message) { bool handled = true; IPC_BEGIN_MESSAGE_MAP(ExtensionHelper, message) IPC_MESSAGE_HANDLER(ExtensionMsg_Response, OnExtensionResponse) IPC_MESSAGE_HANDLER(ExtensionMsg_MessageInvoke, OnExtensionMessageInvoke) IPC_MESSAGE_HANDLER(ExtensionMsg_DispatchOnConnect, OnExtensionDispatchOnConnect) IPC_MESSAGE_HANDLER(ExtensionMsg_DeliverMessage, OnExtensionDeliverMessage) IPC_MESSAGE_HANDLER(ExtensionMsg_DispatchOnDisconnect, OnExtensionDispatchOnDisconnect) IPC_MESSAGE_HANDLER(ExtensionMsg_ExecuteCode, OnExecuteCode) IPC_MESSAGE_HANDLER(ExtensionMsg_GetApplicationInfo, OnGetApplicationInfo) IPC_MESSAGE_HANDLER(ExtensionMsg_SetTabId, OnSetTabId) IPC_MESSAGE_HANDLER(ExtensionMsg_UpdateBrowserWindowId, OnUpdateBrowserWindowId) IPC_MESSAGE_HANDLER(ExtensionMsg_NotifyRenderViewType, OnNotifyRendererViewType) IPC_MESSAGE_HANDLER(ExtensionMsg_AddMessageToConsole, OnAddMessageToConsole) IPC_MESSAGE_HANDLER(ExtensionMsg_AppWindowClosed, OnAppWindowClosed); IPC_MESSAGE_UNHANDLED(handled = false) IPC_END_MESSAGE_MAP() return handled; } void ExtensionHelper::DidFinishDocumentLoad(WebFrame* frame) { dispatcher_->user_script_slave()->InjectScripts( frame, UserScript::DOCUMENT_END); SchedulerMap::iterator i = g_schedulers.Get().find(frame); if (i != g_schedulers.Get().end()) i->second->DidFinishDocumentLoad(); } void ExtensionHelper::DidFinishLoad(WebKit::WebFrame* frame) { SchedulerMap::iterator i = g_schedulers.Get().find(frame); if (i != g_schedulers.Get().end()) i->second->DidFinishLoad(); } void ExtensionHelper::DidCreateDocumentElement(WebFrame* frame) { dispatcher_->user_script_slave()->InjectScripts( frame, UserScript::DOCUMENT_START); SchedulerMap::iterator i = g_schedulers.Get().find(frame); if (i != g_schedulers.Get().end()) i->second->DidCreateDocumentElement(); dispatcher_->DidCreateDocumentElement(frame); } void ExtensionHelper::DidStartProvisionalLoad(WebKit::WebFrame* frame) { SchedulerMap::iterator i = g_schedulers.Get().find(frame); if (i != g_schedulers.Get().end()) i->second->DidStartProvisionalLoad(); } void ExtensionHelper::DraggableRegionsChanged(WebKit::WebFrame* frame) { WebKit::WebVector webregions = frame->document().draggableRegions(); std::vector regions; for (size_t i = 0; i < webregions.size(); ++i) { DraggableRegion region; region.bounds = webregions[i].bounds; region.draggable = webregions[i].draggable; regions.push_back(region); } Send(new ExtensionHostMsg_UpdateDraggableRegions(routing_id(), regions)); } void ExtensionHelper::FrameDetached(WebFrame* frame) { // This could be called before DidCreateDataSource, in which case the frame // won't be in the map. SchedulerMap::iterator i = g_schedulers.Get().find(frame); if (i == g_schedulers.Get().end()) return; delete i->second; g_schedulers.Get().erase(i); } void ExtensionHelper::DidCreateDataSource(WebFrame* frame, WebDataSource* ds) { // If there are any app-related fetches in progress, they can be cancelled now // since we have navigated away from the page that created them. if (!frame->parent()) { app_icon_fetchers_.clear(); app_definition_fetcher_.reset(NULL); } // Check first if we created a scheduler for the frame, since this function // gets called for navigations within the document. if (g_schedulers.Get().count(frame)) return; g_schedulers.Get()[frame] = new UserScriptScheduler(frame, dispatcher_); } void ExtensionHelper::OnExtensionResponse(int request_id, bool success, const base::ListValue& response, const std::string& error) { dispatcher_->OnExtensionResponse(request_id, success, response, error); } void ExtensionHelper::OnExtensionMessageInvoke(const std::string& extension_id, const std::string& function_name, const base::ListValue& args, const GURL& event_url, bool user_gesture) { scoped_ptr web_user_gesture; if (user_gesture) { web_user_gesture.reset(new WebScopedUserGesture); } dispatcher_->v8_context_set().DispatchChromeHiddenMethod( extension_id, function_name, args, render_view(), event_url); } void ExtensionHelper::OnExtensionDispatchOnConnect( int target_port_id, const std::string& channel_name, const std::string& tab_json, const std::string& source_extension_id, const std::string& target_extension_id) { MiscellaneousBindings::DispatchOnConnect( dispatcher_->v8_context_set().GetAll(), target_port_id, channel_name, tab_json, source_extension_id, target_extension_id, render_view()); } void ExtensionHelper::OnExtensionDeliverMessage(int target_id, const std::string& message) { MiscellaneousBindings::DeliverMessage(dispatcher_->v8_context_set().GetAll(), target_id, message, render_view()); } void ExtensionHelper::OnExtensionDispatchOnDisconnect( int port_id, const std::string& error_message) { MiscellaneousBindings::DispatchOnDisconnect( dispatcher_->v8_context_set().GetAll(), port_id, error_message, render_view()); } void ExtensionHelper::OnExecuteCode( const ExtensionMsg_ExecuteCode_Params& params) { WebView* webview = render_view()->GetWebView(); WebFrame* main_frame = webview->mainFrame(); if (!main_frame) { ListValue val; Send(new ExtensionHostMsg_ExecuteCodeFinished( routing_id(), params.request_id, "No main frame", -1, GURL(""), val)); return; } // chrome.tabs.executeScript() only supports execution in either the top frame // or all frames. We handle both cases in the top frame. SchedulerMap::iterator i = g_schedulers.Get().find(main_frame); if (i != g_schedulers.Get().end()) i->second->ExecuteCode(params); } void ExtensionHelper::OnGetApplicationInfo(int page_id) { WebApplicationInfo app_info; if (page_id == render_view()->GetPageId()) { string16 error; web_apps::ParseWebAppFromWebDocument( render_view()->GetWebView()->mainFrame(), &app_info, &error); } // Prune out any data URLs in the set of icons. The browser process expects // any icon with a data URL to have originated from a favicon. We don't want // to decode arbitrary data URLs in the browser process. See // http://b/issue?id=1162972 for (size_t i = 0; i < app_info.icons.size(); ++i) { if (app_info.icons[i].url.SchemeIs(chrome::kDataScheme)) { app_info.icons.erase(app_info.icons.begin() + i); --i; } } Send(new ExtensionHostMsg_DidGetApplicationInfo( routing_id(), page_id, app_info)); } void ExtensionHelper::OnNotifyRendererViewType(ViewType type) { view_type_ = type; } void ExtensionHelper::OnSetTabId(int init_tab_id) { CHECK_EQ(tab_id_, -1); CHECK_GE(init_tab_id, 0); tab_id_ = init_tab_id; } void ExtensionHelper::OnUpdateBrowserWindowId(int window_id) { browser_window_id_ = window_id; } void ExtensionHelper::OnAddMessageToConsole(ConsoleMessageLevel level, const std::string& message) { console::AddMessage(render_view(), level, message); } void ExtensionHelper::OnAppWindowClosed() { v8::HandleScope scope; v8::Handle script_context = render_view()->GetWebView()->mainFrame()->mainWorldScriptContext(); ChromeV8Context* chrome_v8_context = dispatcher_->v8_context_set().GetByV8Context(script_context); if (!chrome_v8_context) return; chrome_v8_context->CallChromeHiddenMethod("OnAppWindowClosed", 0, NULL, NULL); } } // namespace extensions