[test] Move cctest/test-access-check to unittests/
... api/access-check-unittest. Bug: v8:12781 Change-Id: I25025af204cdeaba1ba2b690cbaf5ba93e59441f Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3725447 Commit-Queue: 王澳 <wangao.james@bytedance.com> Reviewed-by: Leszek Swirski <leszeks@chromium.org> Cr-Commit-Position: refs/heads/main@{#81392}
This commit is contained in:
parent
7a2686016d
commit
5c0a114a4f
@ -163,7 +163,6 @@ v8_source_set("cctest_sources") {
|
||||
"scope-test-helper.h",
|
||||
"setup-isolate-for-tests.cc",
|
||||
"setup-isolate-for-tests.h",
|
||||
"test-access-checks.cc",
|
||||
"test-accessor-assembler.cc",
|
||||
"test-accessors.cc",
|
||||
"test-allocation.cc",
|
||||
|
@ -1,462 +0,0 @@
|
||||
// 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 "include/v8-container.h"
|
||||
#include "include/v8-external.h"
|
||||
#include "include/v8-function.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::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::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::ALL_CAN_READ);
|
||||
|
||||
v8::Local<v8::Object> obj = tmpl->NewRemoteInstance().ToLocalChecked();
|
||||
|
||||
v8::Local<v8::Context> context = v8::Context::New(isolate);
|
||||
CheckCrossContextAccess(isolate, context, obj);
|
||||
}
|
@ -11,11 +11,118 @@
|
||||
#include "include/v8-template.h"
|
||||
#include "src/debug/debug.h"
|
||||
#include "test/unittests/test-utils.h"
|
||||
#include "testing/gmock-support.h"
|
||||
#include "testing/gtest/include/gtest/gtest.h"
|
||||
|
||||
namespace v8 {
|
||||
|
||||
using AccessCheckTest = TestWithIsolate;
|
||||
using testing::IsInt32;
|
||||
using testing::IsString;
|
||||
|
||||
int32_t g_cross_context_int = 0;
|
||||
|
||||
bool g_expect_interceptor_call = false;
|
||||
|
||||
class AccessCheckTest : public TestWithIsolate {
|
||||
public:
|
||||
void CheckCanRunScriptInContext(Local<Context> context) {
|
||||
HandleScope handle_scope(isolate());
|
||||
Context::Scope context_scope(context);
|
||||
|
||||
g_expect_interceptor_call = false;
|
||||
g_cross_context_int = 0;
|
||||
|
||||
// Running script in this context should work.
|
||||
RunJS("this.foo = 42; this[23] = true;");
|
||||
EXPECT_THAT(RunJS("this.all_can_read"), IsInt32(42));
|
||||
RunJS("this.cross_context_int = 23");
|
||||
CHECK_EQ(g_cross_context_int, 23);
|
||||
EXPECT_THAT(RunJS("this.cross_context_int"), IsInt32(23));
|
||||
}
|
||||
|
||||
void CheckCrossContextAccess(Local<Context> accessing_context,
|
||||
Local<Object> accessed_object) {
|
||||
HandleScope handle_scope(isolate());
|
||||
accessing_context->Global()
|
||||
->Set(accessing_context, NewString("other"), accessed_object)
|
||||
.FromJust();
|
||||
Context::Scope context_scope(accessing_context);
|
||||
|
||||
g_expect_interceptor_call = true;
|
||||
g_cross_context_int = 23;
|
||||
|
||||
{
|
||||
TryCatch try_catch(isolate());
|
||||
CHECK(TryRunJS("this.other.foo").IsEmpty());
|
||||
}
|
||||
{
|
||||
TryCatch try_catch(isolate());
|
||||
CHECK(TryRunJS("this.other[23]").IsEmpty());
|
||||
}
|
||||
|
||||
// AllCanRead properties are also inaccessible.
|
||||
{
|
||||
TryCatch try_catch(isolate());
|
||||
CHECK(TryRunJS("this.other.all_can_read").IsEmpty());
|
||||
}
|
||||
|
||||
// Intercepted properties are accessible, however.
|
||||
EXPECT_THAT(RunJS("this.other.cross_context_int"), IsInt32(23));
|
||||
RunJS("this.other.cross_context_int = 42");
|
||||
EXPECT_THAT(RunJS("this.other[7]"), IsInt32(42));
|
||||
EXPECT_THAT(RunJS("JSON.stringify(Object.getOwnPropertyNames(this.other))"),
|
||||
IsString("[\"7\",\"cross_context_int\"]"));
|
||||
}
|
||||
|
||||
void CheckCrossContextAccessWithException(Local<Context> accessing_context,
|
||||
Local<Object> accessed_object) {
|
||||
HandleScope handle_scope(isolate());
|
||||
accessing_context->Global()
|
||||
->Set(accessing_context, NewString("other"), accessed_object)
|
||||
.FromJust();
|
||||
Context::Scope context_scope(accessing_context);
|
||||
|
||||
{
|
||||
TryCatch try_catch(isolate());
|
||||
TryRunJS("this.other.should_throw");
|
||||
CHECK(try_catch.HasCaught());
|
||||
CHECK(try_catch.Exception()->IsString());
|
||||
CHECK(NewString("exception")
|
||||
->Equals(accessing_context, try_catch.Exception())
|
||||
.FromJust());
|
||||
}
|
||||
|
||||
{
|
||||
TryCatch try_catch(isolate());
|
||||
TryRunJS("this.other.should_throw = 8");
|
||||
CHECK(try_catch.HasCaught());
|
||||
CHECK(try_catch.Exception()->IsString());
|
||||
CHECK(NewString("exception")
|
||||
->Equals(accessing_context, try_catch.Exception())
|
||||
.FromJust());
|
||||
}
|
||||
|
||||
{
|
||||
TryCatch try_catch(isolate());
|
||||
TryRunJS("this.other[42]");
|
||||
CHECK(try_catch.HasCaught());
|
||||
CHECK(try_catch.Exception()->IsString());
|
||||
CHECK(NewString("exception")
|
||||
->Equals(accessing_context, try_catch.Exception())
|
||||
.FromJust());
|
||||
}
|
||||
|
||||
{
|
||||
TryCatch try_catch(isolate());
|
||||
TryRunJS("this.other[42] = 8");
|
||||
CHECK(try_catch.HasCaught());
|
||||
CHECK(try_catch.Exception()->IsString());
|
||||
CHECK(NewString("exception")
|
||||
->Equals(accessing_context, try_catch.Exception())
|
||||
.FromJust());
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
namespace {
|
||||
|
||||
@ -185,4 +292,330 @@ TEST_F(AccessRegressionTest,
|
||||
ASSERT_EQ(getter_c2->native_context(), *Utils::OpenHandle(*context2));
|
||||
}
|
||||
|
||||
void NamedGetter(Local<Name> property,
|
||||
const PropertyCallbackInfo<Value>& info) {
|
||||
CHECK(g_expect_interceptor_call);
|
||||
Isolate* isolate = info.GetIsolate();
|
||||
Local<Context> context = isolate->GetCurrentContext();
|
||||
if (property
|
||||
->Equals(context, String::NewFromUtf8(isolate, "cross_context_int")
|
||||
.ToLocalChecked())
|
||||
.FromJust())
|
||||
info.GetReturnValue().Set(g_cross_context_int);
|
||||
}
|
||||
|
||||
void NamedSetter(Local<Name> property, Local<Value> value,
|
||||
const PropertyCallbackInfo<Value>& info) {
|
||||
CHECK(g_expect_interceptor_call);
|
||||
Isolate* isolate = info.GetIsolate();
|
||||
Local<Context> context = isolate->GetCurrentContext();
|
||||
if (!property
|
||||
->Equals(context, String::NewFromUtf8(isolate, "cross_context_int")
|
||||
.ToLocalChecked())
|
||||
.FromJust())
|
||||
return;
|
||||
if (value->IsInt32()) {
|
||||
g_cross_context_int = value->ToInt32(context).ToLocalChecked()->Value();
|
||||
}
|
||||
info.GetReturnValue().Set(value);
|
||||
}
|
||||
|
||||
void NamedQuery(Local<Name> property,
|
||||
const PropertyCallbackInfo<Integer>& info) {
|
||||
CHECK(g_expect_interceptor_call);
|
||||
Isolate* isolate = info.GetIsolate();
|
||||
Local<Context> context = isolate->GetCurrentContext();
|
||||
if (!property
|
||||
->Equals(context, String::NewFromUtf8(isolate, "cross_context_int")
|
||||
.ToLocalChecked())
|
||||
.FromJust())
|
||||
return;
|
||||
info.GetReturnValue().Set(DontDelete);
|
||||
}
|
||||
|
||||
void NamedDeleter(Local<Name> property,
|
||||
const PropertyCallbackInfo<Boolean>& info) {
|
||||
CHECK(g_expect_interceptor_call);
|
||||
Isolate* isolate = info.GetIsolate();
|
||||
Local<Context> context = isolate->GetCurrentContext();
|
||||
if (!property
|
||||
->Equals(context, String::NewFromUtf8(isolate, "cross_context_int")
|
||||
.ToLocalChecked())
|
||||
.FromJust())
|
||||
return;
|
||||
info.GetReturnValue().Set(false);
|
||||
}
|
||||
|
||||
void NamedEnumerator(const PropertyCallbackInfo<Array>& info) {
|
||||
CHECK(g_expect_interceptor_call);
|
||||
Isolate* isolate = info.GetIsolate();
|
||||
Local<Context> context = isolate->GetCurrentContext();
|
||||
Local<Array> names = Array::New(isolate, 1);
|
||||
names
|
||||
->Set(context, 0,
|
||||
String::NewFromUtf8(isolate, "cross_context_int").ToLocalChecked())
|
||||
.FromJust();
|
||||
info.GetReturnValue().Set(names);
|
||||
}
|
||||
|
||||
void IndexedGetter(uint32_t index, const PropertyCallbackInfo<Value>& info) {
|
||||
CHECK(g_expect_interceptor_call);
|
||||
if (index == 7) info.GetReturnValue().Set(g_cross_context_int);
|
||||
}
|
||||
|
||||
void IndexedSetter(uint32_t index, Local<Value> value,
|
||||
const PropertyCallbackInfo<Value>& info) {
|
||||
CHECK(g_expect_interceptor_call);
|
||||
Isolate* isolate = info.GetIsolate();
|
||||
Local<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 PropertyCallbackInfo<Integer>& info) {
|
||||
CHECK(g_expect_interceptor_call);
|
||||
if (index == 7) info.GetReturnValue().Set(DontDelete);
|
||||
}
|
||||
|
||||
void IndexedDeleter(uint32_t index, const PropertyCallbackInfo<Boolean>& info) {
|
||||
CHECK(g_expect_interceptor_call);
|
||||
if (index == 7) info.GetReturnValue().Set(false);
|
||||
}
|
||||
|
||||
void IndexedEnumerator(const PropertyCallbackInfo<Array>& info) {
|
||||
CHECK(g_expect_interceptor_call);
|
||||
Isolate* isolate = info.GetIsolate();
|
||||
Local<Context> context = isolate->GetCurrentContext();
|
||||
Local<Array> names = Array::New(isolate, 1);
|
||||
names->Set(context, 0, String::NewFromUtf8(isolate, "7").ToLocalChecked())
|
||||
.FromJust();
|
||||
info.GetReturnValue().Set(names);
|
||||
}
|
||||
|
||||
void MethodGetter(Local<Name> property,
|
||||
const PropertyCallbackInfo<Value>& info) {
|
||||
Isolate* isolate = info.GetIsolate();
|
||||
Local<Context> context = isolate->GetCurrentContext();
|
||||
|
||||
Local<External> data = info.Data().As<External>();
|
||||
Local<FunctionTemplate>& function_template =
|
||||
*reinterpret_cast<Local<FunctionTemplate>*>(data->Value());
|
||||
|
||||
info.GetReturnValue().Set(
|
||||
function_template->GetFunction(context).ToLocalChecked());
|
||||
}
|
||||
|
||||
void MethodCallback(const FunctionCallbackInfo<Value>& info) {
|
||||
info.GetReturnValue().Set(8);
|
||||
}
|
||||
|
||||
void NamedGetterThrowsException(Local<Name> property,
|
||||
const PropertyCallbackInfo<Value>& info) {
|
||||
info.GetIsolate()->ThrowException(
|
||||
String::NewFromUtf8(info.GetIsolate(), "exception").ToLocalChecked());
|
||||
}
|
||||
|
||||
void NamedSetterThrowsException(Local<Name> property, Local<Value> value,
|
||||
const PropertyCallbackInfo<Value>& info) {
|
||||
info.GetIsolate()->ThrowException(
|
||||
String::NewFromUtf8(info.GetIsolate(), "exception").ToLocalChecked());
|
||||
}
|
||||
|
||||
void IndexedGetterThrowsException(uint32_t index,
|
||||
const PropertyCallbackInfo<Value>& info) {
|
||||
info.GetIsolate()->ThrowException(
|
||||
String::NewFromUtf8(info.GetIsolate(), "exception").ToLocalChecked());
|
||||
}
|
||||
|
||||
void IndexedSetterThrowsException(uint32_t index, Local<Value> value,
|
||||
const PropertyCallbackInfo<Value>& info) {
|
||||
info.GetIsolate()->ThrowException(
|
||||
String::NewFromUtf8(info.GetIsolate(), "exception").ToLocalChecked());
|
||||
}
|
||||
|
||||
void GetCrossContextInt(Local<String> property,
|
||||
const PropertyCallbackInfo<Value>& info) {
|
||||
CHECK(!g_expect_interceptor_call);
|
||||
info.GetReturnValue().Set(g_cross_context_int);
|
||||
}
|
||||
|
||||
void SetCrossContextInt(Local<String> property, Local<Value> value,
|
||||
const PropertyCallbackInfo<void>& info) {
|
||||
CHECK(!g_expect_interceptor_call);
|
||||
Isolate* isolate = info.GetIsolate();
|
||||
Local<Context> context = isolate->GetCurrentContext();
|
||||
if (value->IsInt32()) {
|
||||
g_cross_context_int = value->ToInt32(context).ToLocalChecked()->Value();
|
||||
}
|
||||
}
|
||||
|
||||
void Return42(Local<String> property, const PropertyCallbackInfo<Value>& info) {
|
||||
info.GetReturnValue().Set(42);
|
||||
}
|
||||
|
||||
void Ctor(const FunctionCallbackInfo<Value>& info) {
|
||||
CHECK(info.IsConstructCall());
|
||||
}
|
||||
|
||||
TEST_F(AccessCheckTest, AccessCheckWithInterceptor) {
|
||||
Local<ObjectTemplate> global_template = ObjectTemplate::New(isolate());
|
||||
global_template->SetAccessCheckCallbackAndHandler(
|
||||
AccessCheck,
|
||||
NamedPropertyHandlerConfiguration(NamedGetter, NamedSetter, NamedQuery,
|
||||
NamedDeleter, NamedEnumerator),
|
||||
IndexedPropertyHandlerConfiguration(IndexedGetter, IndexedSetter,
|
||||
IndexedQuery, IndexedDeleter,
|
||||
IndexedEnumerator));
|
||||
global_template->SetNativeDataProperty(
|
||||
NewString("cross_context_int"), GetCrossContextInt, SetCrossContextInt);
|
||||
global_template->SetNativeDataProperty(NewString("all_can_read"), Return42,
|
||||
nullptr, Local<Value>(), None,
|
||||
ALL_CAN_READ);
|
||||
|
||||
Local<Context> context0 = Context::New(isolate(), nullptr, global_template);
|
||||
CheckCanRunScriptInContext(context0);
|
||||
|
||||
// Create another context.
|
||||
Local<Context> context1 = Context::New(isolate(), nullptr, global_template);
|
||||
CheckCrossContextAccess(context1, context0->Global());
|
||||
}
|
||||
|
||||
TEST_F(AccessCheckTest, CallFunctionWithRemoteContextReceiver) {
|
||||
HandleScope scope(isolate());
|
||||
Local<FunctionTemplate> global_template = FunctionTemplate::New(isolate());
|
||||
|
||||
Local<Signature> signature = Signature::New(isolate(), global_template);
|
||||
Local<FunctionTemplate> function_template = FunctionTemplate::New(
|
||||
isolate(), MethodCallback, External::New(isolate(), &function_template),
|
||||
signature);
|
||||
|
||||
global_template->InstanceTemplate()->SetAccessCheckCallbackAndHandler(
|
||||
AccessCheck,
|
||||
NamedPropertyHandlerConfiguration(
|
||||
MethodGetter, nullptr, nullptr, nullptr, nullptr,
|
||||
External::New(isolate(), &function_template)),
|
||||
IndexedPropertyHandlerConfiguration());
|
||||
|
||||
Local<Object> accessed_object =
|
||||
Context::NewRemoteContext(isolate(), global_template->InstanceTemplate())
|
||||
.ToLocalChecked();
|
||||
Local<Context> accessing_context =
|
||||
Context::New(isolate(), nullptr, global_template->InstanceTemplate());
|
||||
|
||||
HandleScope handle_scope(isolate());
|
||||
accessing_context->Global()
|
||||
->Set(accessing_context, NewString("other"), accessed_object)
|
||||
.FromJust();
|
||||
Context::Scope context_scope(accessing_context);
|
||||
|
||||
{
|
||||
TryCatch try_catch(isolate());
|
||||
EXPECT_THAT(RunJS("this.other.method()"), IsInt32(8));
|
||||
CHECK(!try_catch.HasCaught());
|
||||
}
|
||||
}
|
||||
|
||||
TEST_F(AccessCheckTest, AccessCheckWithExceptionThrowingInterceptor) {
|
||||
isolate()->SetFailedAccessCheckCallbackFunction(
|
||||
[](Local<Object> target, AccessType type, Local<Value> data) {
|
||||
UNREACHABLE(); // This should never be called.
|
||||
});
|
||||
|
||||
Local<ObjectTemplate> global_template = ObjectTemplate::New(isolate());
|
||||
global_template->SetAccessCheckCallbackAndHandler(
|
||||
AccessCheck,
|
||||
NamedPropertyHandlerConfiguration(NamedGetterThrowsException,
|
||||
NamedSetterThrowsException),
|
||||
IndexedPropertyHandlerConfiguration(IndexedGetterThrowsException,
|
||||
IndexedSetterThrowsException));
|
||||
|
||||
// Create two contexts.
|
||||
Local<Context> context0 = Context::New(isolate(), nullptr, global_template);
|
||||
Local<Context> context1 = Context::New(isolate(), nullptr, global_template);
|
||||
|
||||
CheckCrossContextAccessWithException(context1, context0->Global());
|
||||
}
|
||||
|
||||
TEST_F(AccessCheckTest, NewRemoteContext) {
|
||||
Local<ObjectTemplate> global_template = ObjectTemplate::New(isolate());
|
||||
global_template->SetAccessCheckCallbackAndHandler(
|
||||
AccessCheck,
|
||||
NamedPropertyHandlerConfiguration(NamedGetter, NamedSetter, NamedQuery,
|
||||
NamedDeleter, NamedEnumerator),
|
||||
IndexedPropertyHandlerConfiguration(IndexedGetter, IndexedSetter,
|
||||
IndexedQuery, IndexedDeleter,
|
||||
IndexedEnumerator));
|
||||
global_template->SetNativeDataProperty(
|
||||
NewString("cross_context_int"), GetCrossContextInt, SetCrossContextInt);
|
||||
global_template->SetNativeDataProperty(NewString("all_can_read"), Return42,
|
||||
nullptr, Local<Value>(), None,
|
||||
ALL_CAN_READ);
|
||||
|
||||
Local<Object> global0 =
|
||||
Context::NewRemoteContext(isolate(), global_template).ToLocalChecked();
|
||||
|
||||
// Create a real context.
|
||||
{
|
||||
HandleScope other_scope(isolate());
|
||||
Local<Context> context1 = Context::New(isolate(), nullptr, global_template);
|
||||
|
||||
CheckCrossContextAccess(context1, global0);
|
||||
}
|
||||
|
||||
// Create a context using the detached global.
|
||||
{
|
||||
HandleScope other_scope(isolate());
|
||||
Local<Context> context2 =
|
||||
Context::New(isolate(), nullptr, global_template, global0);
|
||||
|
||||
CheckCanRunScriptInContext(context2);
|
||||
}
|
||||
|
||||
// Turn a regular context into a remote context.
|
||||
{
|
||||
HandleScope other_scope(isolate());
|
||||
Local<Context> context3 = Context::New(isolate(), nullptr, global_template);
|
||||
|
||||
CheckCanRunScriptInContext(context3);
|
||||
|
||||
// Turn the global object into a remote context, and try to access it.
|
||||
Local<Object> context3_global = context3->Global();
|
||||
context3->DetachGlobal();
|
||||
Local<Object> global3 =
|
||||
Context::NewRemoteContext(isolate(), global_template, context3_global)
|
||||
.ToLocalChecked();
|
||||
Local<Context> context4 = Context::New(isolate(), nullptr, global_template);
|
||||
|
||||
CheckCrossContextAccess(context4, global3);
|
||||
|
||||
// Turn it back into a regular context.
|
||||
Local<Context> context5 =
|
||||
Context::New(isolate(), nullptr, global_template, global3);
|
||||
|
||||
CheckCanRunScriptInContext(context5);
|
||||
}
|
||||
}
|
||||
|
||||
TEST_F(AccessCheckTest, NewRemoteInstance) {
|
||||
Local<FunctionTemplate> tmpl = FunctionTemplate::New(isolate(), Ctor);
|
||||
Local<ObjectTemplate> instance = tmpl->InstanceTemplate();
|
||||
instance->SetAccessCheckCallbackAndHandler(
|
||||
AccessCheck,
|
||||
NamedPropertyHandlerConfiguration(NamedGetter, NamedSetter, NamedQuery,
|
||||
NamedDeleter, NamedEnumerator),
|
||||
IndexedPropertyHandlerConfiguration(IndexedGetter, IndexedSetter,
|
||||
IndexedQuery, IndexedDeleter,
|
||||
IndexedEnumerator));
|
||||
tmpl->SetNativeDataProperty(NewString("all_can_read"), Return42, nullptr,
|
||||
Local<Value>(), None, ALL_CAN_READ);
|
||||
|
||||
Local<Object> obj = tmpl->NewRemoteInstance().ToLocalChecked();
|
||||
|
||||
Local<Context> context = Context::New(isolate());
|
||||
CheckCrossContextAccess(context, obj);
|
||||
}
|
||||
|
||||
} // namespace v8
|
||||
|
Loading…
Reference in New Issue
Block a user