blob: 0b1fbc39d6f56e710517bf0d6f648ed206af51fe [file]
// Copyright 2018 The PDFium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "fxjs/cfx_isolate_wrapper.h"
#include <math.h>
#include <memory>
#include "testing/fxv8_unittest.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "v8/include/v8-container.h"
#include "v8/include/v8-context.h"
#include "v8/include/v8-date.h"
#include "v8/include/v8-isolate.h"
namespace {
bool getter_sentinel = false;
bool setter_sentinel = false;
} // namespace
class CFXIsolateWrapperUnitTest : public FXV8UnitTest {
public:
CFXIsolateWrapperUnitTest() = default;
~CFXIsolateWrapperUnitTest() override = default;
// FXV8UnitTest:
void SetUp() override {
FXV8UnitTest::SetUp();
cfx_v8_ = std::make_unique<CFX_IsolateWrapper>(isolate());
}
CFX_IsolateWrapper* cfx_v8() const { return cfx_v8_.get(); }
protected:
std::unique_ptr<CFX_IsolateWrapper> cfx_v8_;
};
TEST_F(CFXIsolateWrapperUnitTest, EmptyLocal) {
v8::Isolate::Scope isolate_scope(isolate());
v8::HandleScope handle_scope(isolate());
v8::Context::Scope context_scope(v8::Context::New(isolate()));
v8::Local<v8::Value> empty;
EXPECT_FALSE(cfx_v8()->ToBooleanReentrant(empty));
EXPECT_EQ(0, cfx_v8()->ToInt32Reentrant(empty));
EXPECT_EQ(0.0, cfx_v8()->ToDoubleReentrant(empty));
EXPECT_EQ("", cfx_v8()->ToByteStringReentrant(empty));
EXPECT_EQ(L"", cfx_v8()->ToWideStringReentrant(empty));
EXPECT_TRUE(cfx_v8()->ToObjectReentrant(empty).IsEmpty());
EXPECT_TRUE(cfx_v8()->ToArrayReentrant(empty).IsEmpty());
// Can't set properties on empty objects, but does not fault.
v8::Local<v8::Value> marker = cfx_v8()->NewNumber(2);
v8::Local<v8::Object> empty_object;
cfx_v8()->PutObjectPropertyReentrant(empty_object, "clams", marker);
EXPECT_TRUE(
cfx_v8()->GetObjectPropertyReentrant(empty_object, "clams").IsEmpty());
EXPECT_EQ(0u, cfx_v8()->GetObjectPropertyNamesReentrant(empty_object).size());
// Can't set elements in empty arrays, but does not fault.
v8::Local<v8::Array> empty_array;
cfx_v8()->PutArrayElementReentrant(empty_array, 0, marker);
EXPECT_TRUE(cfx_v8()->GetArrayElementReentrant(empty_array, 0).IsEmpty());
EXPECT_EQ(0u, cfx_v8()->GetArrayLength(empty_array));
}
TEST_F(CFXIsolateWrapperUnitTest, NewNull) {
v8::Isolate::Scope isolate_scope(isolate());
v8::HandleScope handle_scope(isolate());
v8::Context::Scope context_scope(v8::Context::New(isolate()));
auto nullz = cfx_v8()->NewNull();
EXPECT_FALSE(cfx_v8()->ToBooleanReentrant(nullz));
EXPECT_EQ(0, cfx_v8()->ToInt32Reentrant(nullz));
EXPECT_EQ(0.0, cfx_v8()->ToDoubleReentrant(nullz));
EXPECT_EQ("null", cfx_v8()->ToByteStringReentrant(nullz));
EXPECT_EQ(L"null", cfx_v8()->ToWideStringReentrant(nullz));
EXPECT_TRUE(cfx_v8()->ToObjectReentrant(nullz).IsEmpty());
EXPECT_TRUE(cfx_v8()->ToArrayReentrant(nullz).IsEmpty());
}
TEST_F(CFXIsolateWrapperUnitTest, NewUndefined) {
v8::Isolate::Scope isolate_scope(isolate());
v8::HandleScope handle_scope(isolate());
v8::Context::Scope context_scope(v8::Context::New(isolate()));
auto undef = cfx_v8()->NewUndefined();
EXPECT_FALSE(cfx_v8()->ToBooleanReentrant(undef));
EXPECT_EQ(0, cfx_v8()->ToInt32Reentrant(undef));
EXPECT_TRUE(isnan(cfx_v8()->ToDoubleReentrant(undef)));
EXPECT_EQ("undefined", cfx_v8()->ToByteStringReentrant(undef));
EXPECT_EQ(L"undefined", cfx_v8()->ToWideStringReentrant(undef));
EXPECT_TRUE(cfx_v8()->ToObjectReentrant(undef).IsEmpty());
EXPECT_TRUE(cfx_v8()->ToArrayReentrant(undef).IsEmpty());
}
TEST_F(CFXIsolateWrapperUnitTest, NewBoolean) {
v8::Isolate::Scope isolate_scope(isolate());
v8::HandleScope handle_scope(isolate());
v8::Context::Scope context_scope(v8::Context::New(isolate()));
auto boolz = cfx_v8()->NewBoolean(true);
EXPECT_TRUE(cfx_v8()->ToBooleanReentrant(boolz));
EXPECT_EQ(1, cfx_v8()->ToInt32Reentrant(boolz));
EXPECT_EQ(1.0, cfx_v8()->ToDoubleReentrant(boolz));
EXPECT_EQ("true", cfx_v8()->ToByteStringReentrant(boolz));
EXPECT_EQ(L"true", cfx_v8()->ToWideStringReentrant(boolz));
EXPECT_TRUE(cfx_v8()->ToObjectReentrant(boolz).IsEmpty());
EXPECT_TRUE(cfx_v8()->ToArrayReentrant(boolz).IsEmpty());
boolz = cfx_v8()->NewBoolean(false);
EXPECT_FALSE(cfx_v8()->ToBooleanReentrant(boolz));
EXPECT_EQ(0, cfx_v8()->ToInt32Reentrant(boolz));
EXPECT_EQ(0.0, cfx_v8()->ToDoubleReentrant(boolz));
EXPECT_EQ("false", cfx_v8()->ToByteStringReentrant(boolz));
EXPECT_EQ(L"false", cfx_v8()->ToWideStringReentrant(boolz));
EXPECT_TRUE(cfx_v8()->ToObjectReentrant(boolz).IsEmpty());
EXPECT_TRUE(cfx_v8()->ToArrayReentrant(boolz).IsEmpty());
}
TEST_F(CFXIsolateWrapperUnitTest, NewNumber) {
v8::Isolate::Scope isolate_scope(isolate());
v8::HandleScope handle_scope(isolate());
v8::Context::Scope context_scope(v8::Context::New(isolate()));
auto num = cfx_v8()->NewNumber(42.1);
EXPECT_TRUE(cfx_v8()->ToBooleanReentrant(num));
EXPECT_EQ(42, cfx_v8()->ToInt32Reentrant(num));
EXPECT_EQ(42.1, cfx_v8()->ToDoubleReentrant(num));
EXPECT_EQ("42.1", cfx_v8()->ToByteStringReentrant(num));
EXPECT_EQ(L"42.1", cfx_v8()->ToWideStringReentrant(num));
EXPECT_TRUE(cfx_v8()->ToObjectReentrant(num).IsEmpty());
EXPECT_TRUE(cfx_v8()->ToArrayReentrant(num).IsEmpty());
}
TEST_F(CFXIsolateWrapperUnitTest, NewString) {
v8::Isolate::Scope isolate_scope(isolate());
v8::HandleScope handle_scope(isolate());
v8::Context::Scope context_scope(v8::Context::New(isolate()));
auto str = cfx_v8()->NewString("123");
EXPECT_TRUE(cfx_v8()->ToBooleanReentrant(str));
EXPECT_EQ(123, cfx_v8()->ToInt32Reentrant(str));
EXPECT_EQ(123, cfx_v8()->ToDoubleReentrant(str));
EXPECT_EQ("123", cfx_v8()->ToByteStringReentrant(str));
EXPECT_EQ(L"123", cfx_v8()->ToWideStringReentrant(str));
EXPECT_TRUE(cfx_v8()->ToObjectReentrant(str).IsEmpty());
EXPECT_TRUE(cfx_v8()->ToArrayReentrant(str).IsEmpty());
auto str2 = cfx_v8()->NewString(L"123");
EXPECT_TRUE(cfx_v8()->ToBooleanReentrant(str2));
EXPECT_EQ(123, cfx_v8()->ToInt32Reentrant(str2));
EXPECT_EQ(123, cfx_v8()->ToDoubleReentrant(str2));
EXPECT_EQ("123", cfx_v8()->ToByteStringReentrant(str2));
EXPECT_EQ(L"123", cfx_v8()->ToWideStringReentrant(str2));
EXPECT_TRUE(cfx_v8()->ToObjectReentrant(str2).IsEmpty());
EXPECT_TRUE(cfx_v8()->ToArrayReentrant(str2).IsEmpty());
}
TEST_F(CFXIsolateWrapperUnitTest, NewDate) {
v8::Isolate::Scope isolate_scope(isolate());
v8::HandleScope handle_scope(isolate());
v8::Context::Scope context_scope(v8::Context::New(isolate()));
auto date = cfx_v8()->NewDate(1111111111);
EXPECT_TRUE(cfx_v8()->ToBooleanReentrant(date));
EXPECT_EQ(1111111111, cfx_v8()->ToInt32Reentrant(date));
EXPECT_EQ(1111111111.0, cfx_v8()->ToDoubleReentrant(date));
EXPECT_NE("", cfx_v8()->ToByteStringReentrant(date)); // exact format varies.
EXPECT_NE(L"",
cfx_v8()->ToWideStringReentrant(date)); // exact format varies.
EXPECT_TRUE(cfx_v8()->ToObjectReentrant(date)->IsObject());
EXPECT_TRUE(cfx_v8()->ToArrayReentrant(date).IsEmpty());
}
TEST_F(CFXIsolateWrapperUnitTest, NewArray) {
v8::Isolate::Scope isolate_scope(isolate());
v8::HandleScope handle_scope(isolate());
v8::Context::Scope context_scope(v8::Context::New(isolate()));
auto array = cfx_v8()->NewArray();
EXPECT_EQ(0u, cfx_v8()->GetArrayLength(array));
EXPECT_FALSE(cfx_v8()->GetArrayElementReentrant(array, 2).IsEmpty());
EXPECT_TRUE(cfx_v8()->GetArrayElementReentrant(array, 2)->IsUndefined());
EXPECT_EQ(0u, cfx_v8()->GetArrayLength(array));
cfx_v8()->PutArrayElementReentrant(array, 3, cfx_v8()->NewNumber(12));
EXPECT_FALSE(cfx_v8()->GetArrayElementReentrant(array, 2).IsEmpty());
EXPECT_TRUE(cfx_v8()->GetArrayElementReentrant(array, 2)->IsUndefined());
EXPECT_FALSE(cfx_v8()->GetArrayElementReentrant(array, 3).IsEmpty());
EXPECT_TRUE(cfx_v8()->GetArrayElementReentrant(array, 3)->IsNumber());
EXPECT_EQ(4u, cfx_v8()->GetArrayLength(array));
EXPECT_TRUE(cfx_v8()->ToBooleanReentrant(array));
EXPECT_EQ(0, cfx_v8()->ToInt32Reentrant(array));
double d = cfx_v8()->ToDoubleReentrant(array);
EXPECT_NE(d, d); // i.e. NaN.
EXPECT_EQ(L",,,12", cfx_v8()->ToWideStringReentrant(array));
EXPECT_TRUE(cfx_v8()->ToObjectReentrant(array)->IsObject());
EXPECT_TRUE(cfx_v8()->ToArrayReentrant(array)->IsArray());
}
TEST_F(CFXIsolateWrapperUnitTest, NewObject) {
v8::Isolate::Scope isolate_scope(isolate());
v8::HandleScope handle_scope(isolate());
v8::Context::Scope context_scope(v8::Context::New(isolate()));
auto object = cfx_v8()->NewObject();
ASSERT_FALSE(object.IsEmpty());
EXPECT_EQ(0u, cfx_v8()->GetObjectPropertyNamesReentrant(object).size());
EXPECT_FALSE(cfx_v8()->GetObjectPropertyReentrant(object, "clams").IsEmpty());
EXPECT_TRUE(
cfx_v8()->GetObjectPropertyReentrant(object, "clams")->IsUndefined());
EXPECT_EQ(0u, cfx_v8()->GetObjectPropertyNamesReentrant(object).size());
cfx_v8()->PutObjectPropertyReentrant(object, "clams",
cfx_v8()->NewNumber(12));
EXPECT_FALSE(cfx_v8()->GetObjectPropertyReentrant(object, "clams").IsEmpty());
EXPECT_TRUE(
cfx_v8()->GetObjectPropertyReentrant(object, "clams")->IsNumber());
EXPECT_EQ(1u, cfx_v8()->GetObjectPropertyNamesReentrant(object).size());
EXPECT_EQ(L"clams", cfx_v8()->GetObjectPropertyNamesReentrant(object)[0]);
EXPECT_TRUE(cfx_v8()->ToBooleanReentrant(object));
EXPECT_EQ(0, cfx_v8()->ToInt32Reentrant(object));
double d = cfx_v8()->ToDoubleReentrant(object);
EXPECT_NE(d, d); // i.e. NaN.
EXPECT_EQ(L"[object Object]", cfx_v8()->ToWideStringReentrant(object));
EXPECT_TRUE(cfx_v8()->ToObjectReentrant(object)->IsObject());
EXPECT_TRUE(cfx_v8()->ToArrayReentrant(object).IsEmpty());
}
TEST_F(CFXIsolateWrapperUnitTest, ThrowFromGetter) {
v8::Isolate::Scope isolate_scope(isolate());
v8::HandleScope handle_scope(isolate());
v8::Local<v8::Context> context = v8::Context::New(isolate());
v8::Context::Scope context_scope(context);
v8::Local<v8::Object> object = cfx_v8()->NewObject();
v8::Local<v8::String> name = cfx_v8()->NewString("clams");
EXPECT_TRUE(object
->SetNativeDataProperty(
context, name,
[](v8::Local<v8::Name> property,
const v8::PropertyCallbackInfo<v8::Value>& info) {
getter_sentinel = true;
info.GetIsolate()->ThrowException(property);
})
.FromJust());
getter_sentinel = false;
EXPECT_TRUE(cfx_v8()->GetObjectPropertyReentrant(object, "clams").IsEmpty());
EXPECT_TRUE(getter_sentinel);
}
TEST_F(CFXIsolateWrapperUnitTest, ThrowFromSetter) {
v8::Isolate::Scope isolate_scope(isolate());
v8::HandleScope handle_scope(isolate());
v8::Local<v8::Context> context = v8::Context::New(isolate());
v8::Context::Scope context_scope(context);
v8::Local<v8::Object> object = cfx_v8()->NewObject();
v8::Local<v8::String> name = cfx_v8()->NewString("clams");
EXPECT_TRUE(
object
->SetNativeDataProperty(
context, name, nullptr,
[](v8::Local<v8::Name> property, v8::Local<v8::Value> value,
const v8::PropertyCallbackInfo<v8::Boolean>& info) {
setter_sentinel = true;
info.GetIsolate()->ThrowException(property);
})
.FromJust());
setter_sentinel = false;
cfx_v8()->PutObjectPropertyReentrant(object, "clams", name);
EXPECT_TRUE(setter_sentinel);
}