// Copyright (c) 2017 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "shell/renderer/electron_render_frame_observer.h" #include <string> #include <utility> #include <vector> #include "base/command_line.h" #include "base/strings/string_number_conversions.h" #include "base/trace_event/trace_event.h" #include "content/public/renderer/render_frame.h" #include "content/public/renderer/render_view.h" #include "electron/buildflags/buildflags.h" #include "electron/shell/common/api/api.mojom.h" #include "ipc/ipc_message_macros.h" #include "net/base/net_module.h" #include "net/grit/net_resources.h" #include "services/service_manager/public/cpp/interface_provider.h" #include "shell/common/options_switches.h" #include "shell/common/world_ids.h" #include "third_party/blink/public/platform/web_isolated_world_info.h" #include "third_party/blink/public/web/blink.h" #include "third_party/blink/public/web/web_document.h" #include "third_party/blink/public/web/web_draggable_region.h" #include "third_party/blink/public/web/web_element.h" #include "third_party/blink/public/web/web_local_frame.h" #include "third_party/blink/public/web/web_script_source.h" #include "ui/base/resource/resource_bundle.h" namespace electron { namespace { scoped_refptr<base::RefCountedMemory> NetResourceProvider(int key) { if (key == IDR_DIR_HEADER_HTML) { return ui::ResourceBundle::GetSharedInstance().LoadDataResourceBytes( IDR_DIR_HEADER_HTML); } return nullptr; } } // namespace ElectronRenderFrameObserver::ElectronRenderFrameObserver( content::RenderFrame* frame, RendererClientBase* renderer_client) : content::RenderFrameObserver(frame), render_frame_(frame), renderer_client_(renderer_client) { // Initialise resource for directory listing. net::NetModule::SetResourceProvider(NetResourceProvider); } void ElectronRenderFrameObserver::DidClearWindowObject() { renderer_client_->DidClearWindowObject(render_frame_); } void ElectronRenderFrameObserver::DidInstallConditionalFeatures( v8::Handle<v8::Context> context, int world_id) { if (ShouldNotifyClient(world_id)) renderer_client_->DidCreateScriptContext(context, render_frame_); auto* command_line = base::CommandLine::ForCurrentProcess(); bool use_context_isolation = renderer_client_->isolated_world(); // This logic matches the EXPLAINED logic in electron_renderer_client.cc // to avoid explaining it twice go check that implementation in // DidCreateScriptContext(); bool is_main_world = IsMainWorld(world_id); bool is_main_frame = render_frame_->IsMainFrame(); bool reuse_renderer_processes_enabled = command_line->HasSwitch(switches::kDisableElectronSiteInstanceOverrides); bool is_not_opened = !render_frame_->GetWebFrame()->Opener() || command_line->HasSwitch(switches::kEnableNodeLeakageInRenderers); bool allow_node_in_sub_frames = command_line->HasSwitch(switches::kNodeIntegrationInSubFrames); bool should_create_isolated_context = use_context_isolation && is_main_world && (is_main_frame || allow_node_in_sub_frames) && (is_not_opened || reuse_renderer_processes_enabled); if (should_create_isolated_context) { CreateIsolatedWorldContext(); if (!renderer_client_->IsWebViewFrame(context, render_frame_)) renderer_client_->SetupMainWorldOverrides(context, render_frame_); } #if !BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS) if (world_id >= WorldIDs::ISOLATED_WORLD_ID_EXTENSIONS && world_id <= WorldIDs::ISOLATED_WORLD_ID_EXTENSIONS_END) { renderer_client_->SetupExtensionWorldOverrides(context, render_frame_, world_id); } #endif } void ElectronRenderFrameObserver::DraggableRegionsChanged() { blink::WebVector<blink::WebDraggableRegion> webregions = render_frame_->GetWebFrame()->GetDocument().DraggableRegions(); std::vector<mojom::DraggableRegionPtr> regions; for (auto& webregion : webregions) { auto region = mojom::DraggableRegion::New(); render_frame_->ConvertViewportToWindow(&webregion.bounds); region->bounds = webregion.bounds; region->draggable = webregion.draggable; regions.push_back(std::move(region)); } mojo::Remote<mojom::ElectronBrowser> browser_remote; render_frame_->GetRemoteInterfaces()->GetInterface( browser_remote.BindNewPipeAndPassReceiver()); browser_remote->UpdateDraggableRegions(std::move(regions)); } void ElectronRenderFrameObserver::WillReleaseScriptContext( v8::Local<v8::Context> context, int world_id) { if (ShouldNotifyClient(world_id)) renderer_client_->WillReleaseScriptContext(context, render_frame_); } void ElectronRenderFrameObserver::OnDestruct() { delete this; } void ElectronRenderFrameObserver::CreateIsolatedWorldContext() { auto* frame = render_frame_->GetWebFrame(); blink::WebIsolatedWorldInfo info; // This maps to the name shown in the context combo box in the Console tab // of the dev tools. info.human_readable_name = blink::WebString::FromUTF8("Electron Isolated Context"); // Setup document's origin policy in isolated world info.security_origin = frame->GetDocument().GetSecurityOrigin(); blink::SetIsolatedWorldInfo(WorldIDs::ISOLATED_WORLD_ID, info); // Create initial script context in isolated world blink::WebScriptSource source("void 0"); frame->ExecuteScriptInIsolatedWorld(WorldIDs::ISOLATED_WORLD_ID, source); } bool ElectronRenderFrameObserver::IsMainWorld(int world_id) { return world_id == WorldIDs::MAIN_WORLD_ID; } bool ElectronRenderFrameObserver::IsIsolatedWorld(int world_id) { return world_id == WorldIDs::ISOLATED_WORLD_ID; } bool ElectronRenderFrameObserver::ShouldNotifyClient(int world_id) { bool allow_node_in_sub_frames = base::CommandLine::ForCurrentProcess()->HasSwitch( switches::kNodeIntegrationInSubFrames); if (renderer_client_->isolated_world() && (render_frame_->IsMainFrame() || allow_node_in_sub_frames)) return IsIsolatedWorld(world_id); else return IsMainWorld(world_id); } } // namespace electron