Index: chrome/renderer/extensions/module_system.cc |
diff --git a/chrome/renderer/extensions/module_system.cc b/chrome/renderer/extensions/module_system.cc |
deleted file mode 100644 |
index b050a9a8129edae812b94da4f8bffbf697b5d766..0000000000000000000000000000000000000000 |
--- a/chrome/renderer/extensions/module_system.cc |
+++ /dev/null |
@@ -1,582 +0,0 @@ |
-// 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/module_system.h" |
- |
-#include "base/bind.h" |
-#include "base/command_line.h" |
-#include "base/debug/trace_event.h" |
-#include "base/stl_util.h" |
-#include "base/strings/string_util.h" |
-#include "base/strings/stringprintf.h" |
-#include "chrome/common/chrome_switches.h" |
-#include "chrome/common/extensions/features/feature_channel.h" |
-#include "chrome/renderer/extensions/chrome_v8_context.h" |
-#include "chrome/renderer/extensions/console.h" |
-#include "chrome/renderer/extensions/safe_builtins.h" |
-#include "content/public/renderer/render_view.h" |
-#include "extensions/common/extension_messages.h" |
-#include "third_party/WebKit/public/web/WebFrame.h" |
-#include "third_party/WebKit/public/web/WebScopedMicrotaskSuppression.h" |
- |
-namespace extensions { |
- |
-namespace { |
- |
-const char* kModuleSystem = "module_system"; |
-const char* kModuleName = "module_name"; |
-const char* kModuleField = "module_field"; |
-const char* kModulesField = "modules"; |
- |
-// Logs a fatal error for the calling context, with some added metadata about |
-// the context: |
-// - Its type (blessed, unblessed, etc). |
-// - Whether it's valid. |
-// - The extension ID, if one exists. |
-// |
-// This will only actual be fatal in in dev/canary, since in too many cases |
-// we're at the mercy of the extension or web page's environment. They can mess |
-// up our JS in unexpected ways. Hopefully dev/canary channel will pick up such |
-// problems, but given the wider variety on stable/beta it's impossible to know. |
-void Fatal(ChromeV8Context* context, const std::string& message) { |
- // Prepend some context metadata. |
- std::string full_message = "("; |
- if (!context->is_valid()) |
- full_message += "Invalid "; |
- full_message += context->GetContextTypeDescription(); |
- full_message += " context"; |
- if (context->extension()) { |
- full_message += " for "; |
- full_message += context->extension()->id(); |
- } |
- full_message += ") "; |
- full_message += message; |
- |
- // <= dev means dev, canary, and trunk. |
- if (GetCurrentChannel() <= chrome::VersionInfo::CHANNEL_DEV) |
- console::Fatal(context->isolate()->GetCallingContext(), full_message); |
- else |
- console::Error(context->isolate()->GetCallingContext(), full_message); |
-} |
- |
-void Warn(v8::Isolate* isolate, const std::string& message) { |
- console::Warn(isolate->GetCallingContext(), message); |
-} |
- |
-// Default exception handler which logs the exception. |
-class DefaultExceptionHandler : public ModuleSystem::ExceptionHandler { |
- public: |
- explicit DefaultExceptionHandler(ChromeV8Context* context) |
- : context_(context) {} |
- |
- // Fatally dumps the debug info from |try_catch| to the console. |
- // Make sure this is never used for exceptions that originate in external |
- // code! |
- virtual void HandleUncaughtException(const v8::TryCatch& try_catch) OVERRIDE { |
- v8::HandleScope handle_scope(context_->isolate()); |
- std::string stack_trace = "<stack trace unavailable>"; |
- if (!try_catch.StackTrace().IsEmpty()) { |
- v8::String::Utf8Value stack_value(try_catch.StackTrace()); |
- if (*stack_value) |
- stack_trace.assign(*stack_value, stack_value.length()); |
- else |
- stack_trace = "<could not convert stack trace to string>"; |
- } |
- Fatal(context_, CreateExceptionString(try_catch) + "{" + stack_trace + "}"); |
- } |
- |
- private: |
- ChromeV8Context* context_; |
-}; |
- |
-} // namespace |
- |
-std::string ModuleSystem::ExceptionHandler::CreateExceptionString( |
- const v8::TryCatch& try_catch) { |
- v8::Handle<v8::Message> message(try_catch.Message()); |
- if (message.IsEmpty()) { |
- return "try_catch has no message"; |
- } |
- |
- std::string resource_name = "<unknown resource>"; |
- if (!message->GetScriptResourceName().IsEmpty()) { |
- v8::String::Utf8Value resource_name_v8( |
- message->GetScriptResourceName()->ToString()); |
- resource_name.assign(*resource_name_v8, resource_name_v8.length()); |
- } |
- |
- std::string error_message = "<no error message>"; |
- if (!message->Get().IsEmpty()) { |
- v8::String::Utf8Value error_message_v8(message->Get()); |
- error_message.assign(*error_message_v8, error_message_v8.length()); |
- } |
- |
- return base::StringPrintf("%s:%d: %s", |
- resource_name.c_str(), |
- message->GetLineNumber(), |
- error_message.c_str()); |
-} |
- |
-ModuleSystem::ModuleSystem(ChromeV8Context* context, SourceMap* source_map) |
- : ObjectBackedNativeHandler(context), |
- context_(context), |
- source_map_(source_map), |
- natives_enabled_(0), |
- exception_handler_(new DefaultExceptionHandler(context)) { |
- RouteFunction("require", |
- base::Bind(&ModuleSystem::RequireForJs, base::Unretained(this))); |
- RouteFunction("requireNative", |
- base::Bind(&ModuleSystem::RequireNative, base::Unretained(this))); |
- RouteFunction("privates", |
- base::Bind(&ModuleSystem::Private, base::Unretained(this))); |
- |
- v8::Handle<v8::Object> global(context->v8_context()->Global()); |
- v8::Isolate* isolate = context->isolate(); |
- global->SetHiddenValue( |
- v8::String::NewFromUtf8(isolate, kModulesField), |
- v8::Object::New(isolate)); |
- global->SetHiddenValue( |
- v8::String::NewFromUtf8(isolate, kModuleSystem), |
- v8::External::New(isolate, this)); |
-} |
- |
-ModuleSystem::~ModuleSystem() { |
- Invalidate(); |
-} |
- |
-void ModuleSystem::Invalidate() { |
- if (!is_valid()) |
- return; |
- |
- // Clear the module system properties from the global context. It's polite, |
- // and we use this as a signal in lazy handlers that we no longer exist. |
- { |
- v8::HandleScope scope(GetIsolate()); |
- v8::Handle<v8::Object> global = context()->v8_context()->Global(); |
- global->DeleteHiddenValue( |
- v8::String::NewFromUtf8(GetIsolate(), kModulesField)); |
- global->DeleteHiddenValue( |
- v8::String::NewFromUtf8(GetIsolate(), kModuleSystem)); |
- } |
- |
- // Invalidate all of the successfully required handlers we own. |
- for (NativeHandlerMap::iterator it = native_handler_map_.begin(); |
- it != native_handler_map_.end(); ++it) { |
- it->second->Invalidate(); |
- } |
- |
- ObjectBackedNativeHandler::Invalidate(); |
-} |
- |
-ModuleSystem::NativesEnabledScope::NativesEnabledScope( |
- ModuleSystem* module_system) |
- : module_system_(module_system) { |
- module_system_->natives_enabled_++; |
-} |
- |
-ModuleSystem::NativesEnabledScope::~NativesEnabledScope() { |
- module_system_->natives_enabled_--; |
- CHECK_GE(module_system_->natives_enabled_, 0); |
-} |
- |
-void ModuleSystem::HandleException(const v8::TryCatch& try_catch) { |
- exception_handler_->HandleUncaughtException(try_catch); |
-} |
- |
-v8::Handle<v8::Value> ModuleSystem::Require(const std::string& module_name) { |
- v8::EscapableHandleScope handle_scope(GetIsolate()); |
- return handle_scope.Escape(RequireForJsInner( |
- v8::String::NewFromUtf8(GetIsolate(), module_name.c_str()))); |
-} |
- |
-void ModuleSystem::RequireForJs( |
- const v8::FunctionCallbackInfo<v8::Value>& args) { |
- v8::Handle<v8::String> module_name = args[0]->ToString(); |
- args.GetReturnValue().Set(RequireForJsInner(module_name)); |
-} |
- |
-v8::Local<v8::Value> ModuleSystem::RequireForJsInner( |
- v8::Handle<v8::String> module_name) { |
- v8::EscapableHandleScope handle_scope(GetIsolate()); |
- v8::Context::Scope context_scope(context()->v8_context()); |
- |
- v8::Handle<v8::Object> global(context()->v8_context()->Global()); |
- |
- // The module system might have been deleted. This can happen if a different |
- // context keeps a reference to us, but our frame is destroyed (e.g. |
- // background page keeps reference to chrome object in a closed popup). |
- v8::Handle<v8::Value> modules_value = global->GetHiddenValue( |
- v8::String::NewFromUtf8(GetIsolate(), kModulesField)); |
- if (modules_value.IsEmpty() || modules_value->IsUndefined()) { |
- Warn(GetIsolate(), "Extension view no longer exists"); |
- return v8::Undefined(GetIsolate()); |
- } |
- |
- v8::Handle<v8::Object> modules(v8::Handle<v8::Object>::Cast(modules_value)); |
- v8::Local<v8::Value> exports(modules->Get(module_name)); |
- if (!exports->IsUndefined()) |
- return handle_scope.Escape(exports); |
- |
- std::string module_name_str = *v8::String::Utf8Value(module_name); |
- v8::Handle<v8::Value> source(GetSource(module_name_str)); |
- if (source.IsEmpty() || source->IsUndefined()) { |
- Fatal(context_, "No source for require(" + module_name_str + ")"); |
- return v8::Undefined(GetIsolate()); |
- } |
- v8::Handle<v8::String> wrapped_source(WrapSource( |
- v8::Handle<v8::String>::Cast(source))); |
- // Modules are wrapped in (function(){...}) so they always return functions. |
- v8::Handle<v8::Value> func_as_value = RunString(wrapped_source, module_name); |
- if (func_as_value.IsEmpty() || func_as_value->IsUndefined()) { |
- Fatal(context_, "Bad source for require(" + module_name_str + ")"); |
- return v8::Undefined(GetIsolate()); |
- } |
- |
- v8::Handle<v8::Function> func = v8::Handle<v8::Function>::Cast(func_as_value); |
- |
- exports = v8::Object::New(GetIsolate()); |
- v8::Handle<v8::Object> natives(NewInstance()); |
- CHECK(!natives.IsEmpty()); // this can happen if v8 has issues |
- |
- // These must match the argument order in WrapSource. |
- v8::Handle<v8::Value> args[] = { |
- // CommonJS. |
- natives->Get(v8::String::NewFromUtf8( |
- GetIsolate(), "require", v8::String::kInternalizedString)), |
- natives->Get(v8::String::NewFromUtf8( |
- GetIsolate(), "requireNative", v8::String::kInternalizedString)), |
- exports, |
- // Libraries that we magically expose to every module. |
- console::AsV8Object(), |
- natives->Get(v8::String::NewFromUtf8( |
- GetIsolate(), "privates", v8::String::kInternalizedString)), |
- // Each safe builtin. Keep in order with the arguments in WrapSource. |
- context_->safe_builtins()->GetArray(), |
- context_->safe_builtins()->GetFunction(), |
- context_->safe_builtins()->GetJSON(), |
- context_->safe_builtins()->GetObjekt(), |
- context_->safe_builtins()->GetRegExp(), |
- context_->safe_builtins()->GetString(), }; |
- { |
- v8::TryCatch try_catch; |
- try_catch.SetCaptureMessage(true); |
- context_->CallFunction(func, arraysize(args), args); |
- if (try_catch.HasCaught()) { |
- HandleException(try_catch); |
- return v8::Undefined(GetIsolate()); |
- } |
- } |
- modules->Set(module_name, exports); |
- return handle_scope.Escape(exports); |
-} |
- |
-v8::Local<v8::Value> ModuleSystem::CallModuleMethod( |
- const std::string& module_name, |
- const std::string& method_name) { |
- v8::HandleScope handle_scope(GetIsolate()); |
- v8::Handle<v8::Value> no_args; |
- return CallModuleMethod(module_name, method_name, 0, &no_args); |
-} |
- |
-v8::Local<v8::Value> ModuleSystem::CallModuleMethod( |
- const std::string& module_name, |
- const std::string& method_name, |
- std::vector<v8::Handle<v8::Value> >* args) { |
- return CallModuleMethod( |
- module_name, method_name, args->size(), vector_as_array(args)); |
-} |
- |
-v8::Local<v8::Value> ModuleSystem::CallModuleMethod( |
- const std::string& module_name, |
- const std::string& method_name, |
- int argc, |
- v8::Handle<v8::Value> argv[]) { |
- TRACE_EVENT2("v8", "v8.callModuleMethod", |
- "module_name", module_name, |
- "method_name", method_name); |
- |
- v8::EscapableHandleScope handle_scope(GetIsolate()); |
- v8::Context::Scope context_scope(context()->v8_context()); |
- |
- v8::Local<v8::Value> module; |
- { |
- NativesEnabledScope natives_enabled(this); |
- module = RequireForJsInner( |
- v8::String::NewFromUtf8(GetIsolate(), module_name.c_str())); |
- } |
- |
- if (module.IsEmpty() || !module->IsObject()) { |
- Fatal(context_, |
- "Failed to get module " + module_name + " to call " + method_name); |
- return handle_scope.Escape( |
- v8::Local<v8::Primitive>(v8::Undefined(GetIsolate()))); |
- } |
- |
- v8::Local<v8::Value> value = |
- v8::Handle<v8::Object>::Cast(module)->Get( |
- v8::String::NewFromUtf8(GetIsolate(), method_name.c_str())); |
- if (value.IsEmpty() || !value->IsFunction()) { |
- Fatal(context_, module_name + "." + method_name + " is not a function"); |
- return handle_scope.Escape( |
- v8::Local<v8::Primitive>(v8::Undefined(GetIsolate()))); |
- } |
- |
- v8::Handle<v8::Function> func = v8::Handle<v8::Function>::Cast(value); |
- v8::Local<v8::Value> result; |
- { |
- v8::TryCatch try_catch; |
- try_catch.SetCaptureMessage(true); |
- result = context_->CallFunction(func, argc, argv); |
- if (try_catch.HasCaught()) |
- HandleException(try_catch); |
- } |
- return handle_scope.Escape(result); |
-} |
- |
-void ModuleSystem::RegisterNativeHandler(const std::string& name, |
- scoped_ptr<NativeHandler> native_handler) { |
- native_handler_map_[name] = |
- linked_ptr<NativeHandler>(native_handler.release()); |
-} |
- |
-void ModuleSystem::OverrideNativeHandlerForTest(const std::string& name) { |
- overridden_native_handlers_.insert(name); |
-} |
- |
-void ModuleSystem::RunString(const std::string& code, const std::string& name) { |
- v8::HandleScope handle_scope(GetIsolate()); |
- RunString(v8::String::NewFromUtf8(GetIsolate(), code.c_str()), |
- v8::String::NewFromUtf8(GetIsolate(), name.c_str())); |
-} |
- |
-// static |
-void ModuleSystem::NativeLazyFieldGetter( |
- v8::Local<v8::String> property, |
- const v8::PropertyCallbackInfo<v8::Value>& info) { |
- LazyFieldGetterInner(property, |
- info, |
- &ModuleSystem::RequireNativeFromString); |
-} |
- |
-// static |
-void ModuleSystem::LazyFieldGetter( |
- v8::Local<v8::String> property, |
- const v8::PropertyCallbackInfo<v8::Value>& info) { |
- LazyFieldGetterInner(property, info, &ModuleSystem::Require); |
-} |
- |
-// static |
-void ModuleSystem::LazyFieldGetterInner( |
- v8::Local<v8::String> property, |
- const v8::PropertyCallbackInfo<v8::Value>& info, |
- RequireFunction require_function) { |
- CHECK(!info.Data().IsEmpty()); |
- CHECK(info.Data()->IsObject()); |
- v8::HandleScope handle_scope(info.GetIsolate()); |
- v8::Handle<v8::Object> parameters = v8::Handle<v8::Object>::Cast(info.Data()); |
- // This context should be the same as context()->v8_context(). |
- v8::Handle<v8::Context> context = parameters->CreationContext(); |
- v8::Handle<v8::Object> global(context->Global()); |
- v8::Handle<v8::Value> module_system_value = global->GetHiddenValue( |
- v8::String::NewFromUtf8(info.GetIsolate(), kModuleSystem)); |
- if (module_system_value.IsEmpty() || !module_system_value->IsExternal()) { |
- // ModuleSystem has been deleted. |
- // TODO(kalman): See comment in header file. |
- Warn(info.GetIsolate(), |
- "Module system has been deleted, does extension view exist?"); |
- return; |
- } |
- |
- ModuleSystem* module_system = static_cast<ModuleSystem*>( |
- v8::Handle<v8::External>::Cast(module_system_value)->Value()); |
- |
- std::string name = *v8::String::Utf8Value(parameters->Get( |
- v8::String::NewFromUtf8(info.GetIsolate(), kModuleName))->ToString()); |
- |
- // Switch to our v8 context because we need functions created while running |
- // the require()d module to belong to our context, not the current one. |
- v8::Context::Scope context_scope(context); |
- NativesEnabledScope natives_enabled_scope(module_system); |
- |
- v8::TryCatch try_catch; |
- v8::Handle<v8::Value> module_value = (module_system->*require_function)(name); |
- if (try_catch.HasCaught()) { |
- module_system->HandleException(try_catch); |
- return; |
- } |
- if (module_value.IsEmpty() || !module_value->IsObject()) { |
- // require_function will have already logged this, we don't need to. |
- return; |
- } |
- |
- v8::Handle<v8::Object> module = v8::Handle<v8::Object>::Cast(module_value); |
- v8::Handle<v8::String> field = |
- parameters->Get(v8::String::NewFromUtf8(info.GetIsolate(), kModuleField)) |
- ->ToString(); |
- |
- if (!module->Has(field)) { |
- std::string field_str = *v8::String::Utf8Value(field); |
- Fatal(module_system->context_, |
- "Lazy require of " + name + "." + field_str + " did not set the " + |
- field_str + " field"); |
- return; |
- } |
- |
- v8::Local<v8::Value> new_field = module->Get(field); |
- if (try_catch.HasCaught()) { |
- module_system->HandleException(try_catch); |
- return; |
- } |
- |
- // Ok for it to be undefined, among other things it's how bindings signify |
- // that the extension doesn't have permission to use them. |
- CHECK(!new_field.IsEmpty()); |
- |
- // Delete the getter and set this field to |new_field| so the same object is |
- // returned every time a certain API is accessed. |
- v8::Handle<v8::Object> object = info.This(); |
- object->Delete(property); |
- object->Set(property, new_field); |
- info.GetReturnValue().Set(new_field); |
-} |
- |
-void ModuleSystem::SetLazyField(v8::Handle<v8::Object> object, |
- const std::string& field, |
- const std::string& module_name, |
- const std::string& module_field) { |
- SetLazyField(object, field, module_name, module_field, |
- &ModuleSystem::LazyFieldGetter); |
-} |
- |
-void ModuleSystem::SetLazyField(v8::Handle<v8::Object> object, |
- const std::string& field, |
- const std::string& module_name, |
- const std::string& module_field, |
- v8::AccessorGetterCallback getter) { |
- v8::HandleScope handle_scope(GetIsolate()); |
- v8::Handle<v8::Object> parameters = v8::Object::New(GetIsolate()); |
- parameters->Set(v8::String::NewFromUtf8(GetIsolate(), kModuleName), |
- v8::String::NewFromUtf8(GetIsolate(), module_name.c_str())); |
- parameters->Set(v8::String::NewFromUtf8(GetIsolate(), kModuleField), |
- v8::String::NewFromUtf8(GetIsolate(), module_field.c_str())); |
- object->SetAccessor(v8::String::NewFromUtf8(GetIsolate(), field.c_str()), |
- getter, |
- NULL, |
- parameters); |
-} |
- |
-void ModuleSystem::SetNativeLazyField(v8::Handle<v8::Object> object, |
- const std::string& field, |
- const std::string& module_name, |
- const std::string& module_field) { |
- SetLazyField(object, field, module_name, module_field, |
- &ModuleSystem::NativeLazyFieldGetter); |
-} |
- |
- |
-v8::Handle<v8::Value> ModuleSystem::RunString(v8::Handle<v8::String> code, |
- v8::Handle<v8::String> name) { |
- v8::EscapableHandleScope handle_scope(GetIsolate()); |
- v8::Context::Scope context_scope(context()->v8_context()); |
- |
- // Prepend extensions:: to |name| so that internal code can be differentiated |
- // from external code in stack traces. This has no effect on behaviour. |
- std::string internal_name = base::StringPrintf("extensions::%s", |
- *v8::String::Utf8Value(name)); |
- |
- blink::WebScopedMicrotaskSuppression suppression; |
- v8::TryCatch try_catch; |
- try_catch.SetCaptureMessage(true); |
- v8::Handle<v8::Script> script( |
- v8::Script::Compile(code, |
- v8::String::NewFromUtf8(GetIsolate(), |
- internal_name.c_str(), |
- v8::String::kNormalString, |
- internal_name.size()))); |
- if (try_catch.HasCaught()) { |
- HandleException(try_catch); |
- return v8::Undefined(GetIsolate()); |
- } |
- |
- v8::Local<v8::Value> result = script->Run(); |
- if (try_catch.HasCaught()) { |
- HandleException(try_catch); |
- return v8::Undefined(GetIsolate()); |
- } |
- |
- return handle_scope.Escape(result); |
-} |
- |
-v8::Handle<v8::Value> ModuleSystem::GetSource(const std::string& module_name) { |
- v8::EscapableHandleScope handle_scope(GetIsolate()); |
- if (!source_map_->Contains(module_name)) |
- return v8::Undefined(GetIsolate()); |
- return handle_scope.Escape( |
- v8::Local<v8::Value>(source_map_->GetSource(GetIsolate(), module_name))); |
-} |
- |
-void ModuleSystem::RequireNative( |
- const v8::FunctionCallbackInfo<v8::Value>& args) { |
- CHECK_EQ(1, args.Length()); |
- std::string native_name = *v8::String::Utf8Value(args[0]->ToString()); |
- args.GetReturnValue().Set(RequireNativeFromString(native_name)); |
-} |
- |
-v8::Handle<v8::Value> ModuleSystem::RequireNativeFromString( |
- const std::string& native_name) { |
- if (natives_enabled_ == 0) { |
- // HACK: if in test throw exception so that we can test the natives-disabled |
- // logic; however, under normal circumstances, this is programmer error so |
- // we could crash. |
- if (exception_handler_) { |
- return GetIsolate()->ThrowException( |
- v8::String::NewFromUtf8(GetIsolate(), "Natives disabled")); |
- } |
- Fatal(context_, "Natives disabled for requireNative(" + native_name + ")"); |
- return v8::Undefined(GetIsolate()); |
- } |
- |
- if (overridden_native_handlers_.count(native_name) > 0u) { |
- return RequireForJsInner( |
- v8::String::NewFromUtf8(GetIsolate(), native_name.c_str())); |
- } |
- |
- NativeHandlerMap::iterator i = native_handler_map_.find(native_name); |
- if (i == native_handler_map_.end()) { |
- Fatal(context_, |
- "Couldn't find native for requireNative(" + native_name + ")"); |
- return v8::Undefined(GetIsolate()); |
- } |
- return i->second->NewInstance(); |
-} |
- |
-v8::Handle<v8::String> ModuleSystem::WrapSource(v8::Handle<v8::String> source) { |
- v8::EscapableHandleScope handle_scope(GetIsolate()); |
- // Keep in order with the arguments in RequireForJsInner. |
- v8::Handle<v8::String> left = v8::String::NewFromUtf8( |
- GetIsolate(), |
- "(function(require, requireNative, exports, " |
- "console, privates," |
- "$Array, $Function, $JSON, $Object, $RegExp, $String) {" |
- "'use strict';"); |
- v8::Handle<v8::String> right = v8::String::NewFromUtf8(GetIsolate(), "\n})"); |
- return handle_scope.Escape(v8::Local<v8::String>( |
- v8::String::Concat(left, v8::String::Concat(source, right)))); |
-} |
- |
-void ModuleSystem::Private(const v8::FunctionCallbackInfo<v8::Value>& args) { |
- CHECK_EQ(1, args.Length()); |
- CHECK(args[0]->IsObject()); |
- v8::Local<v8::Object> obj = args[0].As<v8::Object>(); |
- v8::Local<v8::String> privates_key = |
- v8::String::NewFromUtf8(GetIsolate(), "privates"); |
- v8::Local<v8::Value> privates = obj->GetHiddenValue(privates_key); |
- if (privates.IsEmpty()) { |
- privates = v8::Object::New(args.GetIsolate()); |
- obj->SetHiddenValue(privates_key, privates); |
- } |
- args.GetReturnValue().Set(privates); |
-} |
- |
-} // namespace extensions |