blob: e23ead53dfcc82f00e7bea9778714e736862fd4a [file] [log] [blame] [edit]
// 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/chrome_v8_context.h"
#include "base/debug/trace_event.h"
#include "base/logging.h"
#include "base/string_split.h"
#include "base/values.h"
#include "chrome/common/extensions/api/extension_api.h"
#include "chrome/common/extensions/extension.h"
#include "chrome/common/extensions/extension_set.h"
#include "chrome/renderer/extensions/chrome_v8_extension.h"
#include "chrome/renderer/extensions/user_script_slave.h"
#include "content/public/renderer/render_view.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebView.h"
#include "v8/include/v8.h"
namespace extensions {
namespace {
const char kChromeHidden[] = "chromeHidden";
#ifndef NDEBUG
const char kValidateCallbacks[] = "validateCallbacks";
const char kValidateAPI[] = "validateAPI";
#endif
} // namespace
ChromeV8Context::ChromeV8Context(v8::Handle<v8::Context> v8_context,
WebKit::WebFrame* web_frame,
const Extension* extension,
Feature::Context context_type)
: v8_context_(v8::Persistent<v8::Context>::New(v8_context)),
web_frame_(web_frame),
extension_(extension),
context_type_(context_type) {
VLOG(1) << "Created context:\n"
<< " extension id: " << GetExtensionID() << "\n"
<< " frame: " << web_frame_ << "\n"
<< " context type: " << GetContextTypeDescription();
}
ChromeV8Context::~ChromeV8Context() {
VLOG(1) << "Destroyed context for extension\n"
<< " extension id: " << GetExtensionID();
v8_context_.Dispose();
}
std::string ChromeV8Context::GetExtensionID() {
return extension_ ? extension_->id() : "";
}
// static
v8::Handle<v8::Value> ChromeV8Context::GetOrCreateChromeHidden(
v8::Handle<v8::Context> context) {
v8::Local<v8::Object> global = context->Global();
v8::Local<v8::Value> hidden = global->GetHiddenValue(
v8::String::New(kChromeHidden));
if (hidden.IsEmpty() || hidden->IsUndefined()) {
hidden = v8::Object::New();
global->SetHiddenValue(v8::String::New(kChromeHidden), hidden);
#ifndef NDEBUG
// Tell schema_generated_bindings.js to validate callbacks and events
// against their schema definitions.
v8::Local<v8::Object>::Cast(hidden)->Set(
v8::String::New(kValidateCallbacks), v8::True());
// Tell schema_generated_bindings.js to validate API for ambiguity.
v8::Local<v8::Object>::Cast(hidden)->Set(
v8::String::New(kValidateAPI), v8::True());
#endif
}
DCHECK(hidden->IsObject());
return v8::Local<v8::Object>::Cast(hidden);
}
v8::Handle<v8::Value> ChromeV8Context::GetChromeHidden() const {
v8::Local<v8::Object> global = v8_context_->Global();
return global->GetHiddenValue(v8::String::New(kChromeHidden));
}
content::RenderView* ChromeV8Context::GetRenderView() const {
if (web_frame_ && web_frame_->view())
return content::RenderView::FromWebView(web_frame_->view());
else
return NULL;
}
bool ChromeV8Context::CallChromeHiddenMethod(
const std::string& function_name,
int argc,
v8::Handle<v8::Value>* argv,
v8::Handle<v8::Value>* result) const {
v8::Context::Scope context_scope(v8_context_);
// ChromeV8ContextSet calls clear_web_frame() and then schedules a task to
// delete this object. This check prevents a race from attempting to execute
// script on a NULL web_frame_.
if (!web_frame_)
return false;
// Look up the function name, which may be a sub-property like
// "Port.dispatchOnMessage" in the hidden global variable.
v8::Local<v8::Value> value = v8::Local<v8::Value>::New(GetChromeHidden());
if (value.IsEmpty())
return false;
std::vector<std::string> components;
base::SplitStringDontTrim(function_name, '.', &components);
for (size_t i = 0; i < components.size(); ++i) {
if (!value.IsEmpty() && value->IsObject()) {
value = v8::Local<v8::Object>::Cast(value)->Get(
v8::String::New(components[i].c_str()));
}
}
if (value.IsEmpty() || !value->IsFunction()) {
VLOG(1) << "Could not execute chrome hidden method: " << function_name;
return false;
}
TRACE_EVENT1("v8", "v8.callChromeHiddenMethod",
"function_name", function_name);
v8::Local<v8::Function> function = v8::Local<v8::Function>::Cast(value);
v8::Handle<v8::Value> result_temp =
web_frame_->callFunctionEvenIfScriptDisabled(function,
v8::Object::New(),
argc,
argv);
if (result)
*result = result_temp;
return true;
}
const std::set<std::string>& ChromeV8Context::GetAvailableExtensionAPIs() {
if (!available_extension_apis_.get()) {
available_extension_apis_ =
ExtensionAPI::GetSharedInstance()->GetAPIsForContext(
context_type_,
extension_,
UserScriptSlave::GetDataSourceURLForFrame(
web_frame_)).Pass();
}
return *(available_extension_apis_.get());
}
void ChromeV8Context::DispatchOnLoadEvent(bool is_incognito_process,
int manifest_version) {
v8::HandleScope handle_scope;
v8::Handle<v8::Value> argv[] = {
v8::String::New(GetExtensionID().c_str()),
v8::String::New(GetContextTypeDescription().c_str()),
v8::Boolean::New(is_incognito_process),
v8::Integer::New(manifest_version),
};
CallChromeHiddenMethod("dispatchOnLoad", arraysize(argv), argv, NULL);
}
void ChromeV8Context::DispatchOnUnloadEvent() {
v8::HandleScope handle_scope;
CallChromeHiddenMethod("dispatchOnUnload", 0, NULL, NULL);
}
std::string ChromeV8Context::GetContextTypeDescription() {
switch (context_type_) {
case Feature::UNSPECIFIED_CONTEXT: return "UNSPECIFIED";
case Feature::BLESSED_EXTENSION_CONTEXT: return "BLESSED_EXTENSION";
case Feature::UNBLESSED_EXTENSION_CONTEXT: return "UNBLESSED_EXTENSION";
case Feature::CONTENT_SCRIPT_CONTEXT: return "CONTENT_SCRIPT";
case Feature::WEB_PAGE_CONTEXT: return "WEB_PAGE";
}
NOTREACHED();
return "";
}
} // namespace extensions