Skip to content

Commit

Permalink
Merge pull request #848 from serde-rs/num
Browse files Browse the repository at this point in the history
Deserialize small numbers as integers in arbitrary_precision
  • Loading branch information
dtolnay authored Jan 16, 2022
2 parents 6691977 + d541381 commit 36c43bf
Show file tree
Hide file tree
Showing 4 changed files with 84 additions and 10 deletions.
9 changes: 9 additions & 0 deletions src/de.rs
Original file line number Diff line number Diff line change
Expand Up @@ -864,6 +864,15 @@ impl<'de, R: Read<'de>> Deserializer<R> {
buf.push('-');
}
self.scan_integer(&mut buf)?;
if positive {
if let Ok(unsigned) = buf.parse() {
return Ok(ParserNumber::U64(unsigned));
}
} else {
if let Ok(signed) = buf.parse() {
return Ok(ParserNumber::I64(signed));
}
}
Ok(ParserNumber::String(buf))
}

Expand Down
1 change: 1 addition & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -303,6 +303,7 @@
#![doc(html_root_url = "https://docs.rs/serde_json/1.0.74")]
// Ignored clippy lints
#![allow(
clippy::collapsible_else_if,
clippy::comparison_chain,
clippy::deprecated_cfg_attr,
clippy::doc_markdown,
Expand Down
72 changes: 72 additions & 0 deletions tests/regression/issue845.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
use serde::{Deserialize, Deserializer};
use std::convert::TryFrom;
use std::fmt::{self, Display};
use std::marker::PhantomData;
use std::str::FromStr;

pub struct NumberVisitor<T> {
marker: PhantomData<T>,
}

impl<'de, T> serde::de::Visitor<'de> for NumberVisitor<T>
where
T: TryFrom<u64> + TryFrom<i64> + FromStr,
<T as TryFrom<u64>>::Error: Display,
<T as TryFrom<i64>>::Error: Display,
<T as FromStr>::Err: Display,
{
type Value = T;

fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str("an integer or string")
}

fn visit_u64<E>(self, v: u64) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
T::try_from(v).map_err(serde::de::Error::custom)
}

fn visit_i64<E>(self, v: i64) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
T::try_from(v).map_err(serde::de::Error::custom)
}

fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
v.parse().map_err(serde::de::Error::custom)
}
}

fn deserialize_integer_or_string<'de, D, T>(deserializer: D) -> Result<T, D::Error>
where
D: Deserializer<'de>,
T: TryFrom<u64> + TryFrom<i64> + FromStr,
<T as TryFrom<u64>>::Error: Display,
<T as TryFrom<i64>>::Error: Display,
<T as FromStr>::Err: Display,
{
deserializer.deserialize_any(NumberVisitor {
marker: PhantomData,
})
}

#[derive(Deserialize, Debug)]
pub struct Struct {
#[serde(deserialize_with = "deserialize_integer_or_string")]
pub i: i64,
}

#[test]
fn test() {
let j = r#" {"i":100} "#;
println!("{:?}", serde_json::from_str::<Struct>(j).unwrap());

let j = r#" {"i":"100"} "#;
println!("{:?}", serde_json::from_str::<Struct>(j).unwrap());
}
12 changes: 2 additions & 10 deletions tests/test.rs
Original file line number Diff line number Diff line change
Expand Up @@ -715,11 +715,7 @@ fn test_parse_char() {
),
(
"10",
if cfg!(feature = "arbitrary_precision") {
"invalid type: number, expected a character at line 1 column 2"
} else {
"invalid type: integer `10`, expected a character at line 1 column 2"
},
"invalid type: integer `10`, expected a character at line 1 column 2",
),
]);

Expand Down Expand Up @@ -1203,11 +1199,7 @@ fn test_parse_struct() {
test_parse_err::<Outer>(&[
(
"5",
if cfg!(feature = "arbitrary_precision") {
"invalid type: number, expected struct Outer at line 1 column 1"
} else {
"invalid type: integer `5`, expected struct Outer at line 1 column 1"
},
"invalid type: integer `5`, expected struct Outer at line 1 column 1",
),
(
"\"hello\"",
Expand Down

0 comments on commit 36c43bf

Please sign in to comment.