Blink-compatible serialization of dates.
BUG=chromium:148757 Review-Url: https://codereview.chromium.org/2255973005 Cr-Commit-Position: refs/heads/master@{#38795}
This commit is contained in:
parent
8ff52750ed
commit
cff8e03d4d
@ -70,6 +70,8 @@ enum class SerializationTag : uint8_t {
|
||||
kBeginDenseJSArray = 'A',
|
||||
// End of a dense JS array. numProperties:uint32_t length:uint32_t
|
||||
kEndDenseJSArray = '$',
|
||||
// Date. millisSinceEpoch:double
|
||||
kDate = 'D',
|
||||
};
|
||||
|
||||
ValueSerializer::ValueSerializer(Isolate* isolate)
|
||||
@ -268,6 +270,9 @@ Maybe<bool> ValueSerializer::WriteJSReceiver(Handle<JSReceiver> receiver) {
|
||||
case JS_OBJECT_TYPE:
|
||||
case JS_API_OBJECT_TYPE:
|
||||
return WriteJSObject(Handle<JSObject>::cast(receiver));
|
||||
case JS_DATE_TYPE:
|
||||
WriteJSDate(JSDate::cast(*receiver));
|
||||
return Just(true);
|
||||
default:
|
||||
UNIMPLEMENTED();
|
||||
break;
|
||||
@ -353,6 +358,11 @@ Maybe<bool> ValueSerializer::WriteJSArray(Handle<JSArray> array) {
|
||||
return Just(true);
|
||||
}
|
||||
|
||||
void ValueSerializer::WriteJSDate(JSDate* date) {
|
||||
WriteTag(SerializationTag::kDate);
|
||||
WriteDouble(date->value()->Number());
|
||||
}
|
||||
|
||||
Maybe<uint32_t> ValueSerializer::WriteJSObjectProperties(
|
||||
Handle<JSObject> object, Handle<FixedArray> keys) {
|
||||
uint32_t properties_written = 0;
|
||||
@ -533,6 +543,8 @@ MaybeHandle<Object> ValueDeserializer::ReadObject() {
|
||||
return ReadSparseJSArray();
|
||||
case SerializationTag::kBeginDenseJSArray:
|
||||
return ReadDenseJSArray();
|
||||
case SerializationTag::kDate:
|
||||
return ReadJSDate();
|
||||
default:
|
||||
return MaybeHandle<Object>();
|
||||
}
|
||||
@ -661,6 +673,19 @@ MaybeHandle<JSArray> ValueDeserializer::ReadDenseJSArray() {
|
||||
return scope.CloseAndEscape(array);
|
||||
}
|
||||
|
||||
MaybeHandle<JSDate> ValueDeserializer::ReadJSDate() {
|
||||
double value;
|
||||
if (!ReadDouble().To(&value)) return MaybeHandle<JSDate>();
|
||||
uint32_t id = next_id_++;
|
||||
Handle<JSDate> date;
|
||||
if (!JSDate::New(isolate_->date_function(), isolate_->date_function(), value)
|
||||
.ToHandle(&date)) {
|
||||
return MaybeHandle<JSDate>();
|
||||
}
|
||||
AddObjectWithID(id, date);
|
||||
return date;
|
||||
}
|
||||
|
||||
Maybe<uint32_t> ValueDeserializer::ReadJSObjectProperties(
|
||||
Handle<JSObject> object, SerializationTag end_tag) {
|
||||
for (uint32_t num_properties = 0;; num_properties++) {
|
||||
|
@ -20,6 +20,7 @@ namespace internal {
|
||||
|
||||
class HeapNumber;
|
||||
class Isolate;
|
||||
class JSDate;
|
||||
class Object;
|
||||
class Oddball;
|
||||
class Smi;
|
||||
@ -73,6 +74,7 @@ class ValueSerializer {
|
||||
Maybe<bool> WriteJSReceiver(Handle<JSReceiver> receiver) WARN_UNUSED_RESULT;
|
||||
Maybe<bool> WriteJSObject(Handle<JSObject> object) WARN_UNUSED_RESULT;
|
||||
Maybe<bool> WriteJSArray(Handle<JSArray> array) WARN_UNUSED_RESULT;
|
||||
void WriteJSDate(JSDate* date);
|
||||
|
||||
/*
|
||||
* Reads the specified keys from the object and writes key-value pairs to the
|
||||
@ -143,6 +145,7 @@ class ValueDeserializer {
|
||||
MaybeHandle<JSObject> ReadJSObject() WARN_UNUSED_RESULT;
|
||||
MaybeHandle<JSArray> ReadSparseJSArray() WARN_UNUSED_RESULT;
|
||||
MaybeHandle<JSArray> ReadDenseJSArray() WARN_UNUSED_RESULT;
|
||||
MaybeHandle<JSDate> ReadJSDate() WARN_UNUSED_RESULT;
|
||||
|
||||
/*
|
||||
* Reads key-value pairs into the object until the specified end tag is
|
||||
|
@ -1073,5 +1073,58 @@ TEST_F(ValueSerializerTest, DecodeSparseArrayVersion0) {
|
||||
});
|
||||
}
|
||||
|
||||
TEST_F(ValueSerializerTest, RoundTripDate) {
|
||||
RoundTripTest("new Date(1e6)", [this](Local<Value> value) {
|
||||
ASSERT_TRUE(value->IsDate());
|
||||
EXPECT_EQ(1e6, Date::Cast(*value)->ValueOf());
|
||||
EXPECT_TRUE("Object.getPrototypeOf(result) === Date.prototype");
|
||||
});
|
||||
RoundTripTest("new Date(Date.UTC(1867, 6, 1))", [this](Local<Value> value) {
|
||||
ASSERT_TRUE(value->IsDate());
|
||||
EXPECT_TRUE("result.toISOString() === '1867-07-01T00:00:00.000Z'");
|
||||
});
|
||||
RoundTripTest("new Date(NaN)", [this](Local<Value> value) {
|
||||
ASSERT_TRUE(value->IsDate());
|
||||
EXPECT_TRUE(std::isnan(Date::Cast(*value)->ValueOf()));
|
||||
});
|
||||
RoundTripTest(
|
||||
"({ a: new Date(), get b() { return this.a; } })",
|
||||
[this](Local<Value> value) {
|
||||
EXPECT_TRUE(EvaluateScriptForResultBool("result.a instanceof Date"));
|
||||
EXPECT_TRUE(EvaluateScriptForResultBool("result.a === result.b"));
|
||||
});
|
||||
}
|
||||
|
||||
TEST_F(ValueSerializerTest, DecodeDate) {
|
||||
DecodeTest({0xff, 0x09, 0x3f, 0x00, 0x44, 0x00, 0x00, 0x00, 0x00, 0x80, 0x84,
|
||||
0x2e, 0x41, 0x00},
|
||||
[this](Local<Value> value) {
|
||||
ASSERT_TRUE(value->IsDate());
|
||||
EXPECT_EQ(1e6, Date::Cast(*value)->ValueOf());
|
||||
EXPECT_TRUE("Object.getPrototypeOf(result) === Date.prototype");
|
||||
});
|
||||
DecodeTest(
|
||||
{0xff, 0x09, 0x3f, 0x00, 0x44, 0x00, 0x00, 0x20, 0x45, 0x27, 0x89, 0x87,
|
||||
0xc2, 0x00},
|
||||
[this](Local<Value> value) {
|
||||
ASSERT_TRUE(value->IsDate());
|
||||
EXPECT_TRUE("result.toISOString() === '1867-07-01T00:00:00.000Z'");
|
||||
});
|
||||
DecodeTest({0xff, 0x09, 0x3f, 0x00, 0x44, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0xf8, 0x7f, 0x00},
|
||||
[this](Local<Value> value) {
|
||||
ASSERT_TRUE(value->IsDate());
|
||||
EXPECT_TRUE(std::isnan(Date::Cast(*value)->ValueOf()));
|
||||
});
|
||||
DecodeTest(
|
||||
{0xff, 0x09, 0x3f, 0x00, 0x6f, 0x3f, 0x01, 0x53, 0x01, 0x61, 0x3f,
|
||||
0x01, 0x44, 0x00, 0x20, 0x39, 0x50, 0x37, 0x6a, 0x75, 0x42, 0x3f,
|
||||
0x02, 0x53, 0x01, 0x62, 0x3f, 0x02, 0x5e, 0x01, 0x7b, 0x02},
|
||||
[this](Local<Value> value) {
|
||||
EXPECT_TRUE(EvaluateScriptForResultBool("result.a instanceof Date"));
|
||||
EXPECT_TRUE(EvaluateScriptForResultBool("result.a === result.b"));
|
||||
});
|
||||
}
|
||||
|
||||
} // namespace
|
||||
} // namespace v8
|
||||
|
Loading…
Reference in New Issue
Block a user