Home | History | Annotate | Download | only in renderer
      1 // Copyright 2014 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #ifndef EXTENSIONS_RENDERER_DISPATCHER_H_
      6 #define EXTENSIONS_RENDERER_DISPATCHER_H_
      7 
      8 #include <map>
      9 #include <set>
     10 #include <string>
     11 #include <utility>
     12 #include <vector>
     13 
     14 #include "base/scoped_observer.h"
     15 #include "base/timer/timer.h"
     16 #include "content/public/renderer/render_process_observer.h"
     17 #include "extensions/common/event_filter.h"
     18 #include "extensions/common/extension_set.h"
     19 #include "extensions/common/extensions_client.h"
     20 #include "extensions/common/features/feature.h"
     21 #include "extensions/renderer/resource_bundle_source_map.h"
     22 #include "extensions/renderer/script_context.h"
     23 #include "extensions/renderer/script_context_set.h"
     24 #include "extensions/renderer/user_script_set_manager.h"
     25 #include "extensions/renderer/v8_schema_registry.h"
     26 #include "third_party/WebKit/public/platform/WebString.h"
     27 #include "third_party/WebKit/public/platform/WebVector.h"
     28 #include "v8/include/v8.h"
     29 
     30 class ChromeRenderViewTest;
     31 class GURL;
     32 class ModuleSystem;
     33 class URLPattern;
     34 struct ExtensionMsg_ExternalConnectionInfo;
     35 struct ExtensionMsg_Loaded_Params;
     36 struct ExtensionMsg_UpdatePermissions_Params;
     37 
     38 namespace blink {
     39 class WebFrame;
     40 class WebSecurityOrigin;
     41 }
     42 
     43 namespace base {
     44 class DictionaryValue;
     45 class ListValue;
     46 }
     47 
     48 namespace content {
     49 class RenderThread;
     50 }
     51 
     52 namespace extensions {
     53 class ContentWatcher;
     54 class DispatcherDelegate;
     55 class Extension;
     56 class FilteredEventRouter;
     57 class ManifestPermissionSet;
     58 class RequestSender;
     59 class ScriptContext;
     60 class ScriptInjectionManager;
     61 struct Message;
     62 
     63 // Dispatches extension control messages sent to the renderer and stores
     64 // renderer extension related state.
     65 class Dispatcher : public content::RenderProcessObserver,
     66                    public UserScriptSetManager::Observer {
     67  public:
     68   explicit Dispatcher(DispatcherDelegate* delegate);
     69   virtual ~Dispatcher();
     70 
     71   const std::set<std::string>& function_names() const {
     72     return function_names_;
     73   }
     74 
     75   bool is_extension_process() const { return is_extension_process_; }
     76 
     77   const ExtensionSet* extensions() const { return &extensions_; }
     78 
     79   const ScriptContextSet& script_context_set() const {
     80     return script_context_set_;
     81   }
     82 
     83   V8SchemaRegistry* v8_schema_registry() { return v8_schema_registry_.get(); }
     84 
     85   ContentWatcher* content_watcher() { return content_watcher_.get(); }
     86 
     87   RequestSender* request_sender() { return request_sender_.get(); }
     88 
     89   void OnRenderViewCreated(content::RenderView* render_view);
     90 
     91   bool IsExtensionActive(const std::string& extension_id) const;
     92 
     93   // Finds the extension for the JavaScript context associated with the
     94   // specified |frame| and isolated world. If |world_id| is zero, finds the
     95   // extension ID associated with the main world's JavaScript context. If the
     96   // JavaScript context isn't from an extension, returns empty string.
     97   const Extension* GetExtensionFromFrameAndWorld(const blink::WebFrame* frame,
     98                                                  int world_id,
     99                                                  bool use_effective_url);
    100 
    101   void DidCreateScriptContext(blink::WebFrame* frame,
    102                               const v8::Handle<v8::Context>& context,
    103                               int extension_group,
    104                               int world_id);
    105 
    106   void WillReleaseScriptContext(blink::WebFrame* frame,
    107                                 const v8::Handle<v8::Context>& context,
    108                                 int world_id);
    109 
    110   void DidCreateDocumentElement(blink::WebFrame* frame);
    111 
    112   void DidMatchCSS(
    113       blink::WebFrame* frame,
    114       const blink::WebVector<blink::WebString>& newly_matching_selectors,
    115       const blink::WebVector<blink::WebString>& stopped_matching_selectors);
    116 
    117   void OnExtensionResponse(int request_id,
    118                            bool success,
    119                            const base::ListValue& response,
    120                            const std::string& error);
    121 
    122   // Checks that the current context contains an extension that has permission
    123   // to execute the specified function. If it does not, a v8 exception is thrown
    124   // and the method returns false. Otherwise returns true.
    125   bool CheckContextAccessToExtensionAPI(const std::string& function_name,
    126                                         ScriptContext* context) const;
    127 
    128   // Dispatches the event named |event_name| to all render views.
    129   void DispatchEvent(const std::string& extension_id,
    130                      const std::string& event_name) const;
    131 
    132   // Shared implementation of the various MessageInvoke IPCs.
    133   void InvokeModuleSystemMethod(content::RenderView* render_view,
    134                                 const std::string& extension_id,
    135                                 const std::string& module_name,
    136                                 const std::string& function_name,
    137                                 const base::ListValue& args,
    138                                 bool user_gesture);
    139 
    140   void ClearPortData(int port_id);
    141 
    142   // Returns a list of (module name, resource id) pairs for the JS modules to
    143   // add to the source map.
    144   static std::vector<std::pair<std::string, int> > GetJsResources();
    145   static void RegisterNativeHandlers(ModuleSystem* module_system,
    146                                      ScriptContext* context,
    147                                      Dispatcher* dispatcher,
    148                                      RequestSender* request_sender,
    149                                      V8SchemaRegistry* v8_schema_registry);
    150 
    151   bool WasWebRequestUsedBySomeExtensions() const { return webrequest_used_; }
    152 
    153  private:
    154   friend class ::ChromeRenderViewTest;
    155   FRIEND_TEST_ALL_PREFIXES(RendererPermissionsPolicyDelegateTest,
    156                            CannotScriptWebstore);
    157 
    158   // RenderProcessObserver implementation:
    159   virtual bool OnControlMessageReceived(const IPC::Message& message) OVERRIDE;
    160   virtual void WebKitInitialized() OVERRIDE;
    161   virtual void IdleNotification() OVERRIDE;
    162   virtual void OnRenderProcessShutdown() OVERRIDE;
    163 
    164   void OnActivateExtension(const std::string& extension_id);
    165   void OnCancelSuspend(const std::string& extension_id);
    166   void OnClearTabSpecificPermissions(
    167       int tab_id,
    168       const std::vector<std::string>& extension_ids);
    169   void OnDeliverMessage(int target_port_id, const Message& message);
    170   void OnDispatchOnConnect(int target_port_id,
    171                            const std::string& channel_name,
    172                            const base::DictionaryValue& source_tab,
    173                            const ExtensionMsg_ExternalConnectionInfo& info,
    174                            const std::string& tls_channel_id);
    175   void OnDispatchOnDisconnect(int port_id, const std::string& error_message);
    176   void OnLoaded(
    177       const std::vector<ExtensionMsg_Loaded_Params>& loaded_extensions);
    178   void OnLoadedInternal(scoped_refptr<const Extension> extension);
    179   void OnMessageInvoke(const std::string& extension_id,
    180                        const std::string& module_name,
    181                        const std::string& function_name,
    182                        const base::ListValue& args,
    183                        bool user_gesture);
    184   void OnSetChannel(int channel);
    185   void OnSetFunctionNames(const std::vector<std::string>& names);
    186   void OnSetScriptingWhitelist(
    187       const ExtensionsClient::ScriptingWhitelist& extension_ids);
    188   void OnSetSystemFont(const std::string& font_family,
    189                        const std::string& font_size);
    190   void OnShouldSuspend(const std::string& extension_id, uint64 sequence_id);
    191   void OnSuspend(const std::string& extension_id);
    192   void OnTransferBlobs(const std::vector<std::string>& blob_uuids);
    193   void OnUnloaded(const std::string& id);
    194   void OnUpdatePermissions(const ExtensionMsg_UpdatePermissions_Params& params);
    195   void OnUpdateTabSpecificPermissions(const GURL& url,
    196                                       int tab_id,
    197                                       const std::string& extension_id,
    198                                       const URLPatternSet& origin_set);
    199   void OnUsingWebRequestAPI(bool webrequest_used);
    200 
    201   // UserScriptSetManager::Observer implementation.
    202   virtual void OnUserScriptsUpdated(
    203       const std::set<std::string>& changed_extensions,
    204       const std::vector<UserScript*>& scripts) OVERRIDE;
    205 
    206   void UpdateActiveExtensions();
    207 
    208   // Sets up the host permissions for |extension|.
    209   void InitOriginPermissions(const Extension* extension);
    210 
    211   // Updates the host permissions for extension to include only those in
    212   // |new_patterns|, and remove from |old_patterns| that are no longer allowed.
    213   void UpdateOriginPermissions(const Extension* extension,
    214                                const URLPatternSet& old_patterns,
    215                                const URLPatternSet& new_patterns);
    216 
    217   // Enable custom element whitelist in Apps.
    218   void EnableCustomElementWhiteList();
    219 
    220   // Adds or removes bindings for every context belonging to |extension_id|, or
    221   // or all contexts if |extension_id| is empty.
    222   void UpdateBindings(const std::string& extension_id);
    223 
    224   void UpdateBindingsForContext(ScriptContext* context);
    225 
    226   void RegisterBinding(const std::string& api_name, ScriptContext* context);
    227 
    228   void RegisterNativeHandlers(ModuleSystem* module_system,
    229                               ScriptContext* context);
    230 
    231   // Inserts static source code into |source_map_|.
    232   void PopulateSourceMap();
    233 
    234   // Returns whether the current renderer hosts a platform app.
    235   bool IsWithinPlatformApp();
    236 
    237   bool IsSandboxedPage(const GURL& url) const;
    238 
    239   // Returns the Feature::Context type of context for a JavaScript context.
    240   Feature::Context ClassifyJavaScriptContext(
    241       const Extension* extension,
    242       int extension_group,
    243       const GURL& url,
    244       const blink::WebSecurityOrigin& origin);
    245 
    246   // Gets |field| from |object| or creates it as an empty object if it doesn't
    247   // exist.
    248   v8::Handle<v8::Object> GetOrCreateObject(const v8::Handle<v8::Object>& object,
    249                                            const std::string& field,
    250                                            v8::Isolate* isolate);
    251 
    252   v8::Handle<v8::Object> GetOrCreateBindObjectIfAvailable(
    253       const std::string& api_name,
    254       std::string* bind_name,
    255       ScriptContext* context);
    256 
    257   // The delegate for this dispatcher. Not owned, but must extend beyond the
    258   // Dispatcher's own lifetime.
    259   DispatcherDelegate* delegate_;
    260 
    261   // True if this renderer is running extensions.
    262   bool is_extension_process_;
    263 
    264   // Contains all loaded extensions.  This is essentially the renderer
    265   // counterpart to ExtensionService in the browser. It contains information
    266   // about all extensions currently loaded by the browser.
    267   ExtensionSet extensions_;
    268 
    269   // The IDs of extensions that failed to load, mapped to the error message
    270   // generated on failure.
    271   std::map<std::string, std::string> extension_load_errors_;
    272 
    273   // All the bindings contexts that are currently loaded for this renderer.
    274   // There is zero or one for each v8 context.
    275   ScriptContextSet script_context_set_;
    276 
    277   scoped_ptr<ContentWatcher> content_watcher_;
    278 
    279   scoped_ptr<UserScriptSetManager> user_script_set_manager_;
    280 
    281   scoped_ptr<ScriptInjectionManager> script_injection_manager_;
    282 
    283   // Same as above, but on a longer timer and will run even if the process is
    284   // not idle, to ensure that IdleHandle gets called eventually.
    285   scoped_ptr<base::RepeatingTimer<content::RenderThread> > forced_idle_timer_;
    286 
    287   // All declared function names.
    288   std::set<std::string> function_names_;
    289 
    290   // The extensions and apps that are active in this process.
    291   std::set<std::string> active_extension_ids_;
    292 
    293   ResourceBundleSourceMap source_map_;
    294 
    295   // Cache for the v8 representation of extension API schemas.
    296   scoped_ptr<V8SchemaRegistry> v8_schema_registry_;
    297 
    298   // Sends API requests to the extension host.
    299   scoped_ptr<RequestSender> request_sender_;
    300 
    301   // The platforms system font family and size;
    302   std::string system_font_family_;
    303   std::string system_font_size_;
    304 
    305   // Mapping of port IDs to tabs. If there is no tab, the value would be -1.
    306   std::map<int, int> port_to_tab_id_map_;
    307 
    308   // True once WebKit has been initialized (and it is therefore safe to poke).
    309   bool is_webkit_initialized_;
    310 
    311   // It is important for this to come after the ScriptInjectionManager, so that
    312   // the observer is destroyed before the UserScriptSet.
    313   ScopedObserver<UserScriptSetManager, UserScriptSetManager::Observer>
    314       user_script_set_manager_observer_;
    315 
    316   // Status of webrequest usage.
    317   bool webrequest_used_;
    318 
    319   DISALLOW_COPY_AND_ASSIGN(Dispatcher);
    320 };
    321 
    322 }  // namespace extensions
    323 
    324 #endif  // EXTENSIONS_RENDERER_DISPATCHER_H_
    325