blob: acea843c148d8997563cfafce5f7c2a4720dc698 [file] [log] [blame] [edit]
// Copyright 2016 the V8 project 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 <stdlib.h>
#include "test/cctest/cctest.h"
namespace {
int32_t g_cross_context_int = 0;
bool g_expect_interceptor_call = false;
void NamedGetter(v8::Local<v8::Name> property,
const v8::PropertyCallbackInfo<v8::Value>& info) {
CHECK(g_expect_interceptor_call);
v8::Isolate* isolate = info.GetIsolate();
v8::Local<v8::Context> context = isolate->GetCurrentContext();
if (property->Equals(context, v8_str("cross_context_int")).FromJust())
info.GetReturnValue().Set(g_cross_context_int);
}
void NamedSetter(v8::Local<v8::Name> property, v8::Local<v8::Value> value,
const v8::PropertyCallbackInfo<v8::Value>& info) {
CHECK(g_expect_interceptor_call);
v8::Isolate* isolate = info.GetIsolate();
v8::Local<v8::Context> context = isolate->GetCurrentContext();
if (!property->Equals(context, v8_str("cross_context_int")).FromJust())
return;
if (value->IsInt32()) {
g_cross_context_int = value->ToInt32(context).ToLocalChecked()->Value();
}
info.GetReturnValue().Set(value);
}
void NamedQuery(v8::Local<v8::Name> property,
const v8::PropertyCallbackInfo<v8::Integer>& info) {
CHECK(g_expect_interceptor_call);
v8::Isolate* isolate = info.GetIsolate();
v8::Local<v8::Context> context = isolate->GetCurrentContext();
if (!property->Equals(context, v8_str("cross_context_int")).FromJust())
return;
info.GetReturnValue().Set(v8::DontDelete);
}
void NamedDeleter(v8::Local<v8::Name> property,
const v8::PropertyCallbackInfo<v8::Boolean>& info) {
CHECK(g_expect_interceptor_call);
v8::Isolate* isolate = info.GetIsolate();
v8::Local<v8::Context> context = isolate->GetCurrentContext();
if (!property->Equals(context, v8_str("cross_context_int")).FromJust())
return;
info.GetReturnValue().Set(false);
}
void NamedEnumerator(const v8::PropertyCallbackInfo<v8::Array>& info) {
CHECK(g_expect_interceptor_call);
v8::Isolate* isolate = info.GetIsolate();
v8::Local<v8::Context> context = isolate->GetCurrentContext();
v8::Local<v8::Array> names = v8::Array::New(isolate, 1);
names->Set(context, 0, v8_str("cross_context_int")).FromJust();
info.GetReturnValue().Set(names);
}
void IndexedGetter(uint32_t index,
const v8::PropertyCallbackInfo<v8::Value>& info) {
CHECK(g_expect_interceptor_call);
if (index == 7) info.GetReturnValue().Set(g_cross_context_int);
}
void IndexedSetter(uint32_t index, v8::Local<v8::Value> value,
const v8::PropertyCallbackInfo<v8::Value>& info) {
CHECK(g_expect_interceptor_call);
v8::Isolate* isolate = info.GetIsolate();
v8::Local<v8::Context> context = isolate->GetCurrentContext();
if (index != 7) return;
if (value->IsInt32()) {
g_cross_context_int = value->ToInt32(context).ToLocalChecked()->Value();
}
info.GetReturnValue().Set(value);
}
void IndexedQuery(uint32_t index,
const v8::PropertyCallbackInfo<v8::Integer>& info) {
CHECK(g_expect_interceptor_call);
if (index == 7) info.GetReturnValue().Set(v8::DontDelete);
}
void IndexedDeleter(uint32_t index,
const v8::PropertyCallbackInfo<v8::Boolean>& info) {
CHECK(g_expect_interceptor_call);
if (index == 7) info.GetReturnValue().Set(false);
}
void IndexedEnumerator(const v8::PropertyCallbackInfo<v8::Array>& info) {
CHECK(g_expect_interceptor_call);
v8::Isolate* isolate = info.GetIsolate();
v8::Local<v8::Context> context = isolate->GetCurrentContext();
v8::Local<v8::Array> names = v8::Array::New(isolate, 1);
names->Set(context, 0, v8_str("7")).FromJust();
info.GetReturnValue().Set(names);
}
void MethodGetter(v8::Local<v8::Name> property,
const v8::PropertyCallbackInfo<v8::Value>& info) {
v8::Isolate* isolate = info.GetIsolate();
v8::Local<v8::Context> context = isolate->GetCurrentContext();
v8::Local<v8::External> data = info.Data().As<v8::External>();
v8::Local<v8::FunctionTemplate>& function_template =
*reinterpret_cast<v8::Local<v8::FunctionTemplate>*>(data->Value());
info.GetReturnValue().Set(
function_template->GetFunction(context).ToLocalChecked());
}
void MethodCallback(const v8::FunctionCallbackInfo<v8::Value>& info) {
info.GetReturnValue().Set(8);
}
void NamedGetterThrowsException(
v8::Local<v8::Name> property,
const v8::PropertyCallbackInfo<v8::Value>& info) {
info.GetIsolate()->ThrowException(v8_str("exception"));
}
void NamedSetterThrowsException(
v8::Local<v8::Name> property, v8::Local<v8::Value> value,
const v8::PropertyCallbackInfo<v8::Value>& info) {
info.GetIsolate()->ThrowException(v8_str("exception"));
}
void IndexedGetterThrowsException(
uint32_t index, const v8::PropertyCallbackInfo<v8::Value>& info) {
info.GetIsolate()->ThrowException(v8_str("exception"));
}
void IndexedSetterThrowsException(
uint32_t index, v8::Local<v8::Value> value,
const v8::PropertyCallbackInfo<v8::Value>& info) {
info.GetIsolate()->ThrowException(v8_str("exception"));
}
bool AccessCheck(v8::Local<v8::Context> accessing_context,
v8::Local<v8::Object> accessed_object,
v8::Local<v8::Value> data) {
return false;
}
void GetCrossContextInt(v8::Local<v8::String> property,
const v8::PropertyCallbackInfo<v8::Value>& info) {
CHECK(!g_expect_interceptor_call);
info.GetReturnValue().Set(g_cross_context_int);
}
void SetCrossContextInt(v8::Local<v8::String> property,
v8::Local<v8::Value> value,
const v8::PropertyCallbackInfo<void>& info) {
CHECK(!g_expect_interceptor_call);
v8::Isolate* isolate = info.GetIsolate();
v8::Local<v8::Context> context = isolate->GetCurrentContext();
if (value->IsInt32()) {
g_cross_context_int = value->ToInt32(context).ToLocalChecked()->Value();
}
}
void Return42(v8::Local<v8::String> property,
const v8::PropertyCallbackInfo<v8::Value>& info) {
info.GetReturnValue().Set(42);
}
void CheckCanRunScriptInContext(v8::Isolate* isolate,
v8::Local<v8::Context> context) {
v8::HandleScope handle_scope(isolate);
v8::Context::Scope context_scope(context);
g_expect_interceptor_call = false;
g_cross_context_int = 0;
// Running script in this context should work.
CompileRunChecked(isolate, "this.foo = 42; this[23] = true;");
ExpectInt32("this.all_can_read", 42);
CompileRunChecked(isolate, "this.cross_context_int = 23");
CHECK_EQ(g_cross_context_int, 23);
ExpectInt32("this.cross_context_int", 23);
}
void CheckCrossContextAccess(v8::Isolate* isolate,
v8::Local<v8::Context> accessing_context,
v8::Local<v8::Object> accessed_object) {
v8::HandleScope handle_scope(isolate);
accessing_context->Global()
->Set(accessing_context, v8_str("other"), accessed_object)
.FromJust();
v8::Context::Scope context_scope(accessing_context);
g_expect_interceptor_call = true;
g_cross_context_int = 23;
{
v8::TryCatch try_catch(isolate);
CHECK(CompileRun(accessing_context, "this.other.foo").IsEmpty());
}
{
v8::TryCatch try_catch(isolate);
CHECK(CompileRun(accessing_context, "this.other[23]").IsEmpty());
}
// AllCanRead properties are also inaccessible.
{
v8::TryCatch try_catch(isolate);
CHECK(CompileRun(accessing_context, "this.other.all_can_read").IsEmpty());
}
// Intercepted properties are accessible, however.
ExpectInt32("this.other.cross_context_int", 23);
CompileRunChecked(isolate, "this.other.cross_context_int = 42");
ExpectInt32("this.other[7]", 42);
ExpectString("JSON.stringify(Object.getOwnPropertyNames(this.other))",
"[\"7\",\"cross_context_int\"]");
}
void CheckCrossContextAccessWithException(
v8::Isolate* isolate, v8::Local<v8::Context> accessing_context,
v8::Local<v8::Object> accessed_object) {
v8::HandleScope handle_scope(isolate);
accessing_context->Global()
->Set(accessing_context, v8_str("other"), accessed_object)
.FromJust();
v8::Context::Scope context_scope(accessing_context);
{
v8::TryCatch try_catch(isolate);
CompileRun("this.other.should_throw");
CHECK(try_catch.HasCaught());
CHECK(try_catch.Exception()->IsString());
CHECK(v8_str("exception")
->Equals(accessing_context, try_catch.Exception())
.FromJust());
}
{
v8::TryCatch try_catch(isolate);
CompileRun("this.other.should_throw = 8");
CHECK(try_catch.HasCaught());
CHECK(try_catch.Exception()->IsString());
CHECK(v8_str("exception")
->Equals(accessing_context, try_catch.Exception())
.FromJust());
}
{
v8::TryCatch try_catch(isolate);
CompileRun("this.other[42]");
CHECK(try_catch.HasCaught());
CHECK(try_catch.Exception()->IsString());
CHECK(v8_str("exception")
->Equals(accessing_context, try_catch.Exception())
.FromJust());
}
{
v8::TryCatch try_catch(isolate);
CompileRun("this.other[42] = 8");
CHECK(try_catch.HasCaught());
CHECK(try_catch.Exception()->IsString());
CHECK(v8_str("exception")
->Equals(accessing_context, try_catch.Exception())
.FromJust());
}
}
void Ctor(const v8::FunctionCallbackInfo<v8::Value>& info) {
CHECK(info.IsConstructCall());
}
} // namespace
TEST(AccessCheckWithInterceptor) {
v8::Isolate* isolate = CcTest::isolate();
v8::HandleScope scope(isolate);
v8::Local<v8::ObjectTemplate> global_template =
v8::ObjectTemplate::New(isolate);
global_template->SetAccessCheckCallbackAndHandler(
AccessCheck,
v8::NamedPropertyHandlerConfiguration(
NamedGetter, NamedSetter, NamedQuery, NamedDeleter, NamedEnumerator),
v8::IndexedPropertyHandlerConfiguration(IndexedGetter, IndexedSetter,
IndexedQuery, IndexedDeleter,
IndexedEnumerator));
global_template->SetNativeDataProperty(
v8_str("cross_context_int"), GetCrossContextInt, SetCrossContextInt);
global_template->SetNativeDataProperty(
v8_str("all_can_read"), Return42, nullptr, v8::Local<v8::Value>(),
v8::None, v8::Local<v8::AccessorSignature>(), v8::ALL_CAN_READ);
v8::Local<v8::Context> context0 =
v8::Context::New(isolate, nullptr, global_template);
CheckCanRunScriptInContext(isolate, context0);
// Create another context.
v8::Local<v8::Context> context1 =
v8::Context::New(isolate, nullptr, global_template);
CheckCrossContextAccess(isolate, context1, context0->Global());
}
TEST(CallFunctionWithRemoteContextReceiver) {
v8::Isolate* isolate = CcTest::isolate();
v8::HandleScope scope(isolate);
v8::Local<v8::FunctionTemplate> global_template =
v8::FunctionTemplate::New(isolate);
v8::Local<v8::Signature> signature =
v8::Signature::New(isolate, global_template);
v8::Local<v8::FunctionTemplate> function_template = v8::FunctionTemplate::New(
isolate, MethodCallback, v8::External::New(isolate, &function_template),
signature);
global_template->InstanceTemplate()->SetAccessCheckCallbackAndHandler(
AccessCheck, v8::NamedPropertyHandlerConfiguration(
MethodGetter, nullptr, nullptr, nullptr, nullptr,
v8::External::New(isolate, &function_template)),
v8::IndexedPropertyHandlerConfiguration());
v8::Local<v8::Object> accessed_object =
v8::Context::NewRemoteContext(isolate,
global_template->InstanceTemplate())
.ToLocalChecked();
v8::Local<v8::Context> accessing_context =
v8::Context::New(isolate, nullptr, global_template->InstanceTemplate());
v8::HandleScope handle_scope(isolate);
accessing_context->Global()
->Set(accessing_context, v8_str("other"), accessed_object)
.FromJust();
v8::Context::Scope context_scope(accessing_context);
{
v8::TryCatch try_catch(isolate);
ExpectInt32("this.other.method()", 8);
CHECK(!try_catch.HasCaught());
}
}
TEST(AccessCheckWithExceptionThrowingInterceptor) {
v8::Isolate* isolate = CcTest::isolate();
isolate->SetFailedAccessCheckCallbackFunction([](v8::Local<v8::Object> target,
v8::AccessType type,
v8::Local<v8::Value> data) {
UNREACHABLE(); // This should never be called.
});
v8::HandleScope scope(isolate);
v8::Local<v8::ObjectTemplate> global_template =
v8::ObjectTemplate::New(isolate);
global_template->SetAccessCheckCallbackAndHandler(
AccessCheck, v8::NamedPropertyHandlerConfiguration(
NamedGetterThrowsException, NamedSetterThrowsException),
v8::IndexedPropertyHandlerConfiguration(IndexedGetterThrowsException,
IndexedSetterThrowsException));
// Create two contexts.
v8::Local<v8::Context> context0 =
v8::Context::New(isolate, nullptr, global_template);
v8::Local<v8::Context> context1 =
v8::Context::New(isolate, nullptr, global_template);
CheckCrossContextAccessWithException(isolate, context1, context0->Global());
}
TEST(NewRemoteContext) {
v8::Isolate* isolate = CcTest::isolate();
v8::HandleScope scope(isolate);
v8::Local<v8::ObjectTemplate> global_template =
v8::ObjectTemplate::New(isolate);
global_template->SetAccessCheckCallbackAndHandler(
AccessCheck,
v8::NamedPropertyHandlerConfiguration(
NamedGetter, NamedSetter, NamedQuery, NamedDeleter, NamedEnumerator),
v8::IndexedPropertyHandlerConfiguration(IndexedGetter, IndexedSetter,
IndexedQuery, IndexedDeleter,
IndexedEnumerator));
global_template->SetNativeDataProperty(
v8_str("cross_context_int"), GetCrossContextInt, SetCrossContextInt);
global_template->SetNativeDataProperty(
v8_str("all_can_read"), Return42, nullptr, v8::Local<v8::Value>(),
v8::None, v8::Local<v8::AccessorSignature>(), v8::ALL_CAN_READ);
v8::Local<v8::Object> global0 =
v8::Context::NewRemoteContext(isolate, global_template).ToLocalChecked();
// Create a real context.
{
v8::HandleScope other_scope(isolate);
v8::Local<v8::Context> context1 =
v8::Context::New(isolate, nullptr, global_template);
CheckCrossContextAccess(isolate, context1, global0);
}
// Create a context using the detached global.
{
v8::HandleScope other_scope(isolate);
v8::Local<v8::Context> context2 =
v8::Context::New(isolate, nullptr, global_template, global0);
CheckCanRunScriptInContext(isolate, context2);
}
// Turn a regular context into a remote context.
{
v8::HandleScope other_scope(isolate);
v8::Local<v8::Context> context3 =
v8::Context::New(isolate, nullptr, global_template);
CheckCanRunScriptInContext(isolate, context3);
// Turn the global object into a remote context, and try to access it.
v8::Local<v8::Object> context3_global = context3->Global();
context3->DetachGlobal();
v8::Local<v8::Object> global3 =
v8::Context::NewRemoteContext(isolate, global_template, context3_global)
.ToLocalChecked();
v8::Local<v8::Context> context4 =
v8::Context::New(isolate, nullptr, global_template);
CheckCrossContextAccess(isolate, context4, global3);
// Turn it back into a regular context.
v8::Local<v8::Context> context5 =
v8::Context::New(isolate, nullptr, global_template, global3);
CheckCanRunScriptInContext(isolate, context5);
}
}
TEST(NewRemoteInstance) {
v8::Isolate* isolate = CcTest::isolate();
v8::HandleScope scope(isolate);
v8::Local<v8::FunctionTemplate> tmpl =
v8::FunctionTemplate::New(isolate, Ctor);
v8::Local<v8::ObjectTemplate> instance = tmpl->InstanceTemplate();
instance->SetAccessCheckCallbackAndHandler(
AccessCheck,
v8::NamedPropertyHandlerConfiguration(
NamedGetter, NamedSetter, NamedQuery, NamedDeleter, NamedEnumerator),
v8::IndexedPropertyHandlerConfiguration(IndexedGetter, IndexedSetter,
IndexedQuery, IndexedDeleter,
IndexedEnumerator));
tmpl->SetNativeDataProperty(
v8_str("all_can_read"), Return42, nullptr, v8::Local<v8::Value>(),
v8::None, v8::Local<v8::AccessorSignature>(), v8::ALL_CAN_READ);
v8::Local<v8::Object> obj = tmpl->NewRemoteInstance().ToLocalChecked();
v8::Local<v8::Context> context = v8::Context::New(isolate);
CheckCrossContextAccess(isolate, context, obj);
}