// 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. #ifndef EXTENSIONS_RENDERER_DISPATCHER_H_ #define EXTENSIONS_RENDERER_DISPATCHER_H_ #include #include #include #include #include #include "base/scoped_observer.h" #include "base/timer/timer.h" #include "content/public/renderer/render_process_observer.h" #include "extensions/common/event_filter.h" #include "extensions/common/extension_set.h" #include "extensions/common/extensions_client.h" #include "extensions/common/features/feature.h" #include "extensions/renderer/resource_bundle_source_map.h" #include "extensions/renderer/script_context.h" #include "extensions/renderer/script_context_set.h" #include "extensions/renderer/user_script_set_manager.h" #include "extensions/renderer/v8_schema_registry.h" #include "third_party/WebKit/public/platform/WebString.h" #include "third_party/WebKit/public/platform/WebVector.h" #include "v8/include/v8.h" class ChromeRenderViewTest; class GURL; class ModuleSystem; class URLPattern; struct ExtensionMsg_ExternalConnectionInfo; struct ExtensionMsg_Loaded_Params; struct ExtensionMsg_UpdatePermissions_Params; namespace blink { class WebFrame; class WebSecurityOrigin; } namespace base { class DictionaryValue; class ListValue; } namespace content { class RenderThread; } namespace extensions { class ContentWatcher; class DispatcherDelegate; class Extension; class FilteredEventRouter; class ManifestPermissionSet; class RequestSender; class ScriptContext; class ScriptInjectionManager; struct Message; // Dispatches extension control messages sent to the renderer and stores // renderer extension related state. class Dispatcher : public content::RenderProcessObserver, public UserScriptSetManager::Observer { public: explicit Dispatcher(DispatcherDelegate* delegate); virtual ~Dispatcher(); const std::set& function_names() const { return function_names_; } bool is_extension_process() const { return is_extension_process_; } const ExtensionSet* extensions() const { return &extensions_; } const ScriptContextSet& script_context_set() const { return script_context_set_; } V8SchemaRegistry* v8_schema_registry() { return v8_schema_registry_.get(); } ContentWatcher* content_watcher() { return content_watcher_.get(); } RequestSender* request_sender() { return request_sender_.get(); } void OnRenderViewCreated(content::RenderView* render_view); bool IsExtensionActive(const std::string& extension_id) const; // Finds the extension ID for the JavaScript context associated with the // specified |frame| and isolated world. If |world_id| is zero, finds the // extension ID associated with the main world's JavaScript context. If the // JavaScript context isn't from an extension, returns empty string. std::string GetExtensionID(const blink::WebFrame* frame, int world_id); void DidCreateScriptContext(blink::WebFrame* frame, const v8::Handle& context, int extension_group, int world_id); void WillReleaseScriptContext(blink::WebFrame* frame, const v8::Handle& context, int world_id); void DidCreateDocumentElement(blink::WebFrame* frame); void DidMatchCSS( blink::WebFrame* frame, const blink::WebVector& newly_matching_selectors, const blink::WebVector& stopped_matching_selectors); void OnExtensionResponse(int request_id, bool success, const base::ListValue& response, const std::string& error); // Checks that the current context contains an extension that has permission // to execute the specified function. If it does not, a v8 exception is thrown // and the method returns false. Otherwise returns true. bool CheckContextAccessToExtensionAPI(const std::string& function_name, ScriptContext* context) const; // Dispatches the event named |event_name| to all render views. void DispatchEvent(const std::string& extension_id, const std::string& event_name) const; // Shared implementation of the various MessageInvoke IPCs. void InvokeModuleSystemMethod(content::RenderView* render_view, const std::string& extension_id, const std::string& module_name, const std::string& function_name, const base::ListValue& args, bool user_gesture); void ClearPortData(int port_id); // Returns a list of (module name, resource id) pairs for the JS modules to // add to the source map. static std::vector > GetJsResources(); static void RegisterNativeHandlers(ModuleSystem* module_system, ScriptContext* context, Dispatcher* dispatcher, RequestSender* request_sender, V8SchemaRegistry* v8_schema_registry); private: friend class ::ChromeRenderViewTest; FRIEND_TEST_ALL_PREFIXES(RendererPermissionsPolicyDelegateTest, CannotScriptWebstore); // RenderProcessObserver implementation: virtual bool OnControlMessageReceived(const IPC::Message& message) OVERRIDE; virtual void WebKitInitialized() OVERRIDE; virtual void IdleNotification() OVERRIDE; virtual void OnRenderProcessShutdown() OVERRIDE; void OnActivateExtension(const std::string& extension_id); void OnCancelSuspend(const std::string& extension_id); void OnClearTabSpecificPermissions( int tab_id, const std::vector& extension_ids); void OnDeliverMessage(int target_port_id, const Message& message); void OnDispatchOnConnect(int target_port_id, const std::string& channel_name, const base::DictionaryValue& source_tab, const ExtensionMsg_ExternalConnectionInfo& info, const std::string& tls_channel_id); void OnDispatchOnDisconnect(int port_id, const std::string& error_message); void OnLoaded( const std::vector& loaded_extensions); void OnLoadedInternal(scoped_refptr extension); void OnMessageInvoke(const std::string& extension_id, const std::string& module_name, const std::string& function_name, const base::ListValue& args, bool user_gesture); void OnSetChannel(int channel); void OnSetFunctionNames(const std::vector& names); void OnSetScriptingWhitelist( const ExtensionsClient::ScriptingWhitelist& extension_ids); void OnSetSystemFont(const std::string& font_family, const std::string& font_size); void OnShouldSuspend(const std::string& extension_id, uint64 sequence_id); void OnSuspend(const std::string& extension_id); void OnTransferBlobs(const std::vector& blob_uuids); void OnUnloaded(const std::string& id); void OnUpdatePermissions(const ExtensionMsg_UpdatePermissions_Params& params); void OnUpdateTabSpecificPermissions(const GURL& url, int tab_id, const std::string& extension_id, const URLPatternSet& origin_set); void OnUsingWebRequestAPI(bool webrequest_used); // UserScriptSetManager::Observer implementation. virtual void OnUserScriptsUpdated( const std::set& changed_extensions, const std::vector& scripts) OVERRIDE; void UpdateActiveExtensions(); // Sets up the host permissions for |extension|. void InitOriginPermissions(const Extension* extension); // Updates the host permissions for extension to include only those in // |new_patterns|, and remove from |old_patterns| that are no longer allowed. void UpdateOriginPermissions(const Extension* extension, const URLPatternSet& old_patterns, const URLPatternSet& new_patterns); // Enable custom element whitelist in Apps. void EnableCustomElementWhiteList(); // Adds or removes bindings for every context belonging to |extension_id|, or // or all contexts if |extension_id| is empty. void UpdateBindings(const std::string& extension_id); void UpdateBindingsForContext(ScriptContext* context); void RegisterBinding(const std::string& api_name, ScriptContext* context); void RegisterNativeHandlers(ModuleSystem* module_system, ScriptContext* context); // Inserts static source code into |source_map_|. void PopulateSourceMap(); // Returns whether the current renderer hosts a platform app. bool IsWithinPlatformApp(); bool IsSandboxedPage(const GURL& url) const; // Returns the Feature::Context type of context for a JavaScript context. Feature::Context ClassifyJavaScriptContext( const Extension* extension, int extension_group, const GURL& url, const blink::WebSecurityOrigin& origin); // Gets |field| from |object| or creates it as an empty object if it doesn't // exist. v8::Handle GetOrCreateObject(const v8::Handle& object, const std::string& field, v8::Isolate* isolate); v8::Handle GetOrCreateBindObjectIfAvailable( const std::string& api_name, std::string* bind_name, ScriptContext* context); // The delegate for this dispatcher. Not owned, but must extend beyond the // Dispatcher's own lifetime. DispatcherDelegate* delegate_; // True if this renderer is running extensions. bool is_extension_process_; // Contains all loaded extensions. This is essentially the renderer // counterpart to ExtensionService in the browser. It contains information // about all extensions currently loaded by the browser. ExtensionSet extensions_; // The IDs of extensions that failed to load, mapped to the error message // generated on failure. std::map extension_load_errors_; // All the bindings contexts that are currently loaded for this renderer. // There is zero or one for each v8 context. ScriptContextSet script_context_set_; scoped_ptr content_watcher_; scoped_ptr user_script_set_manager_; scoped_ptr script_injection_manager_; // Same as above, but on a longer timer and will run even if the process is // not idle, to ensure that IdleHandle gets called eventually. scoped_ptr > forced_idle_timer_; // All declared function names. std::set function_names_; // The extensions and apps that are active in this process. std::set active_extension_ids_; ResourceBundleSourceMap source_map_; // Cache for the v8 representation of extension API schemas. scoped_ptr v8_schema_registry_; // Sends API requests to the extension host. scoped_ptr request_sender_; // The platforms system font family and size; std::string system_font_family_; std::string system_font_size_; // Mapping of port IDs to tabs. If there is no tab, the value would be -1. std::map port_to_tab_id_map_; // True once WebKit has been initialized (and it is therefore safe to poke). bool is_webkit_initialized_; // It is important for this to come after the ScriptInjectionManager, so that // the observer is destroyed before the UserScriptSet. ScopedObserver user_script_set_manager_observer_; DISALLOW_COPY_AND_ASSIGN(Dispatcher); }; } // namespace extensions #endif // EXTENSIONS_RENDERER_DISPATCHER_H_