use std::collections::BTreeSet;
use serde::de::{self, DeserializeSeed, Error as DeError, MapAccess, SeqAccess, Visitor};
use serde::Deserializer;
use serde_json::{Number, Value};
use crate::error::JcsError;
use crate::MAX_NESTING_DEPTH;
pub const MAX_SAFE_INTEGER: i64 = 9_007_199_254_740_991;
const DEPTH_EXCEEDED_SENTINEL: &str = "nesting depth exceeded maximum of ";
pub fn parse_json_value_no_duplicates(json: &[u8]) -> Result<Value, JcsError> {
let mut deserializer = serde_json::Deserializer::from_slice(json);
deserializer.disable_recursion_limit();
let value = deserialize_json_value_no_duplicates(&mut deserializer).map_err(|e| {
if e.to_string().starts_with(DEPTH_EXCEEDED_SENTINEL) {
JcsError::NestingDepthExceeded
} else {
JcsError::Json(e)
}
})?;
deserializer.end()?;
Ok(value)
}
pub fn deserialize_json_value_no_duplicates<'de, D>(deserializer: D) -> Result<Value, D::Error>
where
D: Deserializer<'de>,
{
NoDuplicateValueSeed { depth: 0 }.deserialize(deserializer)
}
pub fn validate_string_contents(value: &str, context: &str) -> Result<(), String> {
if let Some(ch) = value.chars().find(|&ch| is_noncharacter(ch)) {
return Err(format!(
"{context} contains the forbidden noncharacter U+{:04X}",
ch as u32
));
}
Ok(())
}
#[must_use]
pub const fn is_safe_integer(value: i64) -> bool {
value >= -MAX_SAFE_INTEGER && value <= MAX_SAFE_INTEGER
}
const fn is_noncharacter(ch: char) -> bool {
let code = ch as u32;
(0xFDD0 <= code && code <= 0xFDEF) || code & 0xFFFE == 0xFFFE
}
struct NoDuplicateValueSeed {
depth: usize,
}
impl<'de> DeserializeSeed<'de> for NoDuplicateValueSeed {
type Value = Value;
fn deserialize<D>(self, deserializer: D) -> Result<Self::Value, D::Error>
where
D: Deserializer<'de>,
{
if self.depth > MAX_NESTING_DEPTH {
return Err(D::Error::custom(format!(
"{DEPTH_EXCEEDED_SENTINEL}{MAX_NESTING_DEPTH}"
)));
}
deserializer.deserialize_any(NoDuplicateValueVisitor { depth: self.depth })
}
}
struct NoDuplicateValueVisitor {
depth: usize,
}
impl<'de> Visitor<'de> for NoDuplicateValueVisitor {
type Value = Value;
fn expecting(&self, formatter: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
formatter.write_str("a valid JSON value")
}
fn visit_bool<E>(self, value: bool) -> Result<Self::Value, E> {
Ok(Value::Bool(value))
}
fn visit_i64<E>(self, value: i64) -> Result<Self::Value, E> {
Ok(Value::Number(Number::from(value)))
}
fn visit_u64<E>(self, value: u64) -> Result<Self::Value, E> {
Ok(Value::Number(Number::from(value)))
}
fn visit_f64<E>(self, value: f64) -> Result<Self::Value, E>
where
E: de::Error,
{
Number::from_f64(value)
.map(Value::Number)
.ok_or_else(|| E::custom("encountered a non-finite floating-point number"))
}
fn visit_str<E>(self, value: &str) -> Result<Self::Value, E>
where
E: de::Error,
{
validate_string_contents(value, "string value").map_err(E::custom)?;
Ok(Value::String(value.to_owned()))
}
fn visit_borrowed_str<E>(self, value: &'de str) -> Result<Self::Value, E>
where
E: de::Error,
{
self.visit_str(value)
}
fn visit_string<E>(self, value: String) -> Result<Self::Value, E>
where
E: de::Error,
{
validate_string_contents(&value, "string value").map_err(E::custom)?;
Ok(Value::String(value))
}
fn visit_none<E>(self) -> Result<Self::Value, E> {
Ok(Value::Null)
}
fn visit_unit<E>(self) -> Result<Self::Value, E> {
Ok(Value::Null)
}
fn visit_seq<A>(self, mut access: A) -> Result<Self::Value, A::Error>
where
A: SeqAccess<'de>,
{
let mut values = Vec::with_capacity(access.size_hint().unwrap_or(0));
while let Some(value) = access.next_element_seed(NoDuplicateValueSeed {
depth: self.depth + 1,
})? {
values.push(value);
}
Ok(Value::Array(values))
}
fn visit_map<A>(self, mut access: A) -> Result<Self::Value, A::Error>
where
A: MapAccess<'de>,
{
let Some(first_key) = access.next_key::<String>()? else {
return Ok(Value::Object(serde_json::Map::new()));
};
if !first_key.starts_with('$') {
validate_string_contents(&first_key, "object property name")
.map_err(A::Error::custom)?;
}
let first_value = access.next_value_seed(NoDuplicateValueSeed {
depth: self.depth + 1,
})?;
let mut object = serde_json::Map::new();
object.insert(first_key.clone(), first_value);
let mut seen = BTreeSet::new();
seen.insert(first_key);
while let Some(key) = access.next_key::<String>()? {
if !key.starts_with('$') {
validate_string_contents(&key, "object property name").map_err(A::Error::custom)?;
}
if !seen.insert(key.clone()) {
return Err(A::Error::custom(format!("duplicate property name `{key}`")));
}
let value = access.next_value_seed(NoDuplicateValueSeed {
depth: self.depth + 1,
})?;
object.insert(key, value);
}
serde_json::from_value(Value::Object(object)).map_err(A::Error::custom)
}
}