// Copyright 2004-present Facebook. All Rights Reserved. #include #include #include #include #ifdef WITH_FBJSCEXTENSION #undef ASSERT #include #include "OpaqueJSString.h" #endif #include using namespace facebook; using namespace facebook::react; #ifdef ANDROID #include void prepare() { ALooper_prepare(0); } #else void prepare() { } #endif TEST(Value, Undefined) { prepare(); JSGlobalContextRef ctx = JSGlobalContextCreateInGroup(nullptr, nullptr); Value v(ctx, JSValueMakeUndefined(ctx)); auto s = react::String::adopt(JSValueToStringCopy(ctx, v, nullptr)); EXPECT_EQ("undefined", s.str()); JSGlobalContextRelease(ctx); } TEST(Value, FromJSON) { prepare(); JSGlobalContextRef ctx = JSGlobalContextCreateInGroup(nullptr, nullptr); react::String s("{\"a\": 4}"); Value v(Value::fromJSON(ctx, s)); EXPECT_TRUE(JSValueIsObject(ctx, v)); JSGlobalContextRelease(ctx); } TEST(Value, ToJSONString) { prepare(); JSGlobalContextRef ctx = JSGlobalContextCreateInGroup(nullptr, nullptr); react::String s("{\"a\": 4}"); Value v(Value::fromJSON(ctx, s)); folly::dynamic dyn = folly::parseJson(v.toJSONString()); ASSERT_NE(nullptr, dyn); EXPECT_TRUE(dyn.isObject()); auto val = dyn.at("a"); ASSERT_NE(nullptr, val); ASSERT_TRUE(val.isInt()); EXPECT_EQ(4, val.getInt()); EXPECT_EQ(4.0f, val.asDouble()); JSGlobalContextRelease(ctx); } #ifdef WITH_FBJSCEXTENSION // Just test that handling invalid data doesn't crash. TEST(Value, FromBadUtf8) { prepare(); JSGlobalContextRef ctx = JSGlobalContextCreateInGroup(nullptr, nullptr); // 110xxxxx 10xxxxxx auto dyn = folly::dynamic("\xC0"); Value::fromDynamic(ctx, dyn); dyn = folly::dynamic("\xC0\x00"); Value::fromDynamic(ctx, dyn); // 1110xxxx 10xxxxxx 10xxxxxx dyn = "\xE0"; Value::fromDynamic(ctx, dyn); Value(ctx, Value::fromDynamic(ctx, dyn)).toJSONString(); dyn = "\xE0\x00"; Value::fromDynamic(ctx, dyn); Value(ctx, Value::fromDynamic(ctx, dyn)).toJSONString(); dyn = "\xE0\x00\x00"; Value::fromDynamic(ctx, dyn); Value(ctx, Value::fromDynamic(ctx, dyn)).toJSONString(); dyn = "\xE0\xA0\x00"; Value::fromDynamic(ctx, dyn); // 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx dyn = "\xF0"; Value::fromDynamic(ctx, dyn); Value(ctx, Value::fromDynamic(ctx, dyn)).toJSONString(); dyn = "\xF0\x00\x00\x00"; Value::fromDynamic(ctx, dyn); dyn = "\xF0\x80\x80\x00"; Value::fromDynamic(ctx, dyn); Value(ctx, Value::fromDynamic(ctx, dyn)).toJSONString(); JSGlobalContextRelease(ctx); } // Just test that handling invalid data doesn't crash. TEST(Value, BadUtf16) { prepare(); JSGlobalContextRef ctx = JSGlobalContextCreateInGroup(nullptr, nullptr); UChar buf[] = { 0xDD00, 0xDD00, 0xDD00, 0x1111 }; JSStringRef ref = OpaqueJSString::create(buf, 4).leakRef(); Value v(ctx, ref); v.toJSONString(0); JSGlobalContextRelease(ctx); } #endif