// Copyright (c) 2013 GitHub, Inc. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "renderer/atom_renderer_client.h" #include #include "base/command_line.h" #include "common/node_bindings.h" #include "common/options_switches.h" #include "renderer/api/atom_renderer_bindings.h" #include "renderer/atom_render_view_observer.h" #include "third_party/WebKit/public/web/WebDocument.h" #include "third_party/WebKit/public/web/WebFrame.h" #include "common/v8/node_common.h" namespace atom { namespace { // Security tokens. const char* kSecurityAll = "all"; const char* kSecurityExceptIframe = "except-iframe"; const char* kSecurityManualEnableIframe = "manual-enable-iframe"; const char* kSecurityDisable = "disable"; const char* kSecurityEnableNodeIntegration = "enable-node-integration"; // Scheme used by devtools const char* kChromeDevToolsScheme = "chrome-devtools"; } // namespace AtomRendererClient::AtomRendererClient() : node_integration_(EXCEPT_IFRAME), main_frame_(NULL) { // Translate the token. std::string token = CommandLine::ForCurrentProcess()-> GetSwitchValueASCII(switches::kNodeIntegration); if (token == kSecurityExceptIframe) node_integration_ = EXCEPT_IFRAME; else if (token == kSecurityManualEnableIframe) node_integration_ = MANUAL_ENABLE_IFRAME; else if (token == kSecurityDisable) node_integration_ = DISABLE; else if (token == kSecurityAll) node_integration_ = ALL; if (IsNodeBindingEnabled()) { node_bindings_.reset(NodeBindings::Create(false)); atom_bindings_.reset(new AtomRendererBindings); } } AtomRendererClient::~AtomRendererClient() { } void AtomRendererClient::RenderThreadStarted() { if (!IsNodeBindingEnabled()) return; node_bindings_->Initialize(); node_bindings_->PrepareMessageLoop(); DCHECK(!global_env); // Create a default empty environment which would be used when we need to // run V8 code out of a window context (like running a uv callback). v8::HandleScope handle_scope(node_isolate); v8::Local context = v8::Context::New(node_isolate); global_env = node::Environment::New(context); } void AtomRendererClient::RenderViewCreated(content::RenderView* render_view) { new AtomRenderViewObserver(render_view, this); } void AtomRendererClient::DidCreateScriptContext(WebKit::WebFrame* frame, v8::Handle context, int extension_group, int world_id) { // The first web frame is the main frame. if (main_frame_ == NULL) main_frame_ = frame; if (!IsNodeBindingEnabled(frame)) return; v8::Context::Scope scope(context); // Check the existance of process object to prevent duplicate initialization. if (context->Global()->Has(v8::String::New("process"))) return; // Give the node loop a run to make sure everything is ready. node_bindings_->RunMessageLoop(); // Setup node environment for each window. node::Environment* env = node_bindings_->CreateEnvironment(context); // Add atom-shell extended APIs. atom_bindings_->BindToFrame(frame); // Store the created environment. web_page_envs_.push_back(env); // Make uv loop being wrapped by window context. if (node_bindings_->uv_env() == NULL) node_bindings_->set_uv_env(env); } void AtomRendererClient::WillReleaseScriptContext( WebKit::WebFrame* frame, v8::Handle context, int world_id) { if (!IsNodeBindingEnabled(frame)) return; node::Environment* env = node::Environment::GetCurrent(context); if (env == NULL) { LOG(ERROR) << "Encounter a non-node context when releasing script context"; return; } // Clear the environment. web_page_envs_.erase( std::remove(web_page_envs_.begin(), web_page_envs_.end(), env), web_page_envs_.end()); // Notice that we are not disposing the environment object here, because there // may still be pending uv operations in the uv loop, and when they got done // they would be needing the original environment. // So we are leaking the environment object here, just like Chrome leaking the // memory :) . Since it's only leaked when refreshing or unloading, so as long // as we make sure renderer process is restared then the memory would not be // leaked. // env->Dispose(); // Wrap the uv loop with another environment. if (env == node_bindings_->uv_env()) { node::Environment* env = web_page_envs_.size() > 0 ? web_page_envs_[0] : NULL; node_bindings_->set_uv_env(env); } } bool AtomRendererClient::ShouldFork(WebKit::WebFrame* frame, const GURL& url, const std::string& http_method, bool is_initial_navigation, bool is_server_redirect, bool* send_referrer) { // Handle all the navigations and reloads in browser. return true; } bool AtomRendererClient::IsNodeBindingEnabled(WebKit::WebFrame* frame) { if (node_integration_ == DISABLE) return false; // Do not pollute devtools. else if (frame != NULL && GURL(frame->document().url()).SchemeIs(kChromeDevToolsScheme)) return false; // Node integration is enabled in main frame unless explictly disabled. else if (frame == main_frame_) return true; else if (node_integration_ == MANUAL_ENABLE_IFRAME && frame != NULL && frame->uniqueName().utf8().find(kSecurityEnableNodeIntegration) == std::string::npos) return false; else if (node_integration_ == EXCEPT_IFRAME && frame != NULL) return false; else return true; } } // namespace atom