summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDarshan Sen <raisinten@gmail.com>2022-10-03 01:48:31 +0530
committerGitHub <noreply@github.com>2022-10-02 22:18:31 +0200
commitbac3a1210f9c773112f3d5001305d1cc6a097798 (patch)
tree5b80211a315fba2392d0a9d8bb4811f868333358
parente07b62d74a451a0ba6617f1bc7875d9c71c85ee2 (diff)
fix(serde_v8): serialize objects with numeric keys correctly (#15946)
Signed-off-by: Darshan Sen <raisinten@gmail.com>
-rw-r--r--core/runtime.rs84
-rw-r--r--serde_v8/de.rs8
-rw-r--r--serde_v8/tests/de.rs23
3 files changed, 113 insertions, 2 deletions
diff --git a/core/runtime.rs b/core/runtime.rs
index c88f91d91..df2cd9a97 100644
--- a/core/runtime.rs
+++ b/core/runtime.rs
@@ -3217,6 +3217,90 @@ assertEquals(1, notify_return_value);
}
#[tokio::test]
+ async fn test_sync_op_serialize_object_with_numbers_as_keys() {
+ #[op]
+ fn op_sync_serialize_object_with_numbers_as_keys(
+ value: serde_json::Value,
+ ) -> Result<(), Error> {
+ assert_eq!(
+ value.to_string(),
+ r#"{"lines":{"100":{"unit":"m"},"200":{"unit":"cm"}}}"#
+ );
+ Ok(())
+ }
+
+ let extension = Extension::builder()
+ .ops(vec![op_sync_serialize_object_with_numbers_as_keys::decl()])
+ .build();
+
+ let mut runtime = JsRuntime::new(RuntimeOptions {
+ extensions: vec![extension],
+ ..Default::default()
+ });
+
+ runtime
+ .execute_script(
+ "op_sync_serialize_object_with_numbers_as_keys.js",
+ r#"
+Deno.core.ops.op_sync_serialize_object_with_numbers_as_keys({
+ lines: {
+ 100: {
+ unit: "m"
+ },
+ 200: {
+ unit: "cm"
+ }
+ }
+})
+"#,
+ )
+ .unwrap();
+ runtime.run_event_loop(false).await.unwrap();
+ }
+
+ #[tokio::test]
+ async fn test_async_op_serialize_object_with_numbers_as_keys() {
+ #[op]
+ async fn op_async_serialize_object_with_numbers_as_keys(
+ value: serde_json::Value,
+ ) -> Result<(), Error> {
+ assert_eq!(
+ value.to_string(),
+ r#"{"lines":{"100":{"unit":"m"},"200":{"unit":"cm"}}}"#
+ );
+ Ok(())
+ }
+
+ let extension = Extension::builder()
+ .ops(vec![op_async_serialize_object_with_numbers_as_keys::decl()])
+ .build();
+
+ let mut runtime = JsRuntime::new(RuntimeOptions {
+ extensions: vec![extension],
+ ..Default::default()
+ });
+
+ runtime
+ .execute_script(
+ "op_async_serialize_object_with_numbers_as_keys.js",
+ r#"
+Deno.core.opAsync('op_async_serialize_object_with_numbers_as_keys', {
+ lines: {
+ 100: {
+ unit: "m"
+ },
+ 200: {
+ unit: "cm"
+ }
+ }
+})
+"#,
+ )
+ .unwrap();
+ runtime.run_event_loop(false).await.unwrap();
+ }
+
+ #[tokio::test]
async fn test_set_macrotask_callback_set_next_tick_callback() {
#[op]
async fn op_async_sleep() -> Result<(), Error> {
diff --git a/serde_v8/de.rs b/serde_v8/de.rs
index 79fb9ae6d..dff29b8f9 100644
--- a/serde_v8/de.rs
+++ b/serde_v8/de.rs
@@ -325,8 +325,12 @@ impl<'de, 'a, 'b, 's, 'x> de::Deserializer<'de>
};
visitor.visit_map(map)
} else {
- let prop_names =
- obj.get_own_property_names(self.scope, Default::default());
+ let prop_names = obj.get_own_property_names(
+ self.scope,
+ v8::GetPropertyNamesArgsBuilder::new()
+ .key_conversion(v8::KeyConversionMode::ConvertToString)
+ .build(),
+ );
let keys: Vec<magic::Value> = match prop_names {
Some(names) => from_v8(self.scope, names.into()).unwrap(),
None => vec![],
diff --git a/serde_v8/tests/de.rs b/serde_v8/tests/de.rs
index 7affe5f8e..c67b4012c 100644
--- a/serde_v8/tests/de.rs
+++ b/serde_v8/tests/de.rs
@@ -185,6 +185,29 @@ fn de_map() {
}
#[test]
+fn de_obj_with_numeric_keys() {
+ dedo(
+ r#"({
+ lines: {
+ 100: {
+ unit: "m"
+ },
+ 200: {
+ unit: "cm"
+ }
+ }
+})"#,
+ |scope, v| {
+ let json: serde_json::Value = serde_v8::from_v8(scope, v).unwrap();
+ assert_eq!(
+ json.to_string(),
+ r#"{"lines":{"100":{"unit":"m"},"200":{"unit":"cm"}}}"#
+ );
+ },
+ )
+}
+
+#[test]
fn de_string_or_buffer() {
dedo("'hello'", |scope, v| {
let sob: serde_v8::StringOrBuffer = serde_v8::from_v8(scope, v).unwrap();