mod cmp;
mod from;
mod serialize;
use crate::value::{ValueTrait, ValueType};
use crate::{stry, unlikely, Deserializer, ErrorType, Result};
use halfbrown::HashMap;
use std::fmt;
use std::ops::Index;
#[deprecated(since = "0.1.21", note = "Please use Object instead")]
pub type Map = Object;
pub type Object = HashMap<String, Value>;
pub fn to_value(s: &mut [u8]) -> Result<Value> {
let de = stry!(Deserializer::from_slice(s));
OwnedDeserializer::from_deserializer(de).parse()
}
#[derive(Debug, Clone)]
pub enum Value {
Null,
Bool(bool),
F64(f64),
I64(i64),
String(String),
Array(Vec<Value>),
Object(Object),
}
impl ValueTrait for Value {
type Object = Object;
type Array = Vec<Self>;
fn get(&self, k: &str) -> Option<&Self> {
self.as_object().and_then(|a| a.get(k))
}
fn get_mut(&mut self, k: &str) -> Option<&mut Self> {
self.as_object_mut().and_then(|a| a.get_mut(k))
}
fn get_idx(&self, i: usize) -> Option<&Self> {
self.as_array().and_then(|a| a.get(i))
}
fn get_idx_mut(&mut self, i: usize) -> Option<&mut Self> {
self.as_array_mut().and_then(|a| a.get_mut(i))
}
fn value_type(&self) -> ValueType {
match self {
Self::Null => ValueType::Null,
Self::Bool(_) => ValueType::Bool,
Self::F64(_) => ValueType::F64,
Self::I64(_) => ValueType::I64,
Self::String(_) => ValueType::String,
Self::Array(_) => ValueType::Array,
Self::Object(_) => ValueType::Object,
}
}
fn is_null(&self) -> bool {
match self {
Self::Null => true,
_ => false,
}
}
fn as_bool(&self) -> Option<bool> {
match self {
Self::Bool(b) => Some(*b),
_ => None,
}
}
fn as_i64(&self) -> Option<i64> {
match self {
Self::I64(i) => Some(*i),
_ => None,
}
}
fn as_u64(&self) -> Option<u64> {
#[allow(clippy::cast_sign_loss)]
match self {
Self::I64(i) if *i >= 0 => Some(*i as u64),
_ => None,
}
}
fn as_f64(&self) -> Option<f64> {
match self {
Self::F64(i) => Some(*i),
_ => None,
}
}
fn cast_f64(&self) -> Option<f64> {
#[allow(clippy::cast_precision_loss)]
match self {
Self::F64(i) => Some(*i),
Self::I64(i) => Some(*i as f64),
_ => None,
}
}
fn as_string(&self) -> Option<String> {
match self {
Self::String(s) => Some(s.clone()),
_ => None,
}
}
fn as_str(&self) -> Option<&str> {
match self {
Self::String(s) => Some(s.as_str()),
_ => None,
}
}
fn as_array(&self) -> Option<&Vec<Self>> {
match self {
Self::Array(a) => Some(a),
_ => None,
}
}
fn as_array_mut(&mut self) -> Option<&mut Vec<Self>> {
match self {
Self::Array(a) => Some(a),
_ => None,
}
}
fn as_object(&self) -> Option<&<Self as ValueTrait>::Object> {
match self {
Self::Object(m) => Some(m),
_ => None,
}
}
fn as_object_mut(&mut self) -> Option<&mut <Self as ValueTrait>::Object> {
match self {
Self::Object(m) => Some(m),
_ => None,
}
}
}
impl fmt::Display for Value {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match self {
Self::Null => f.write_str("null"),
Self::Bool(false) => f.write_str("false"),
Self::Bool(true) => f.write_str("true"),
Self::I64(n) => f.write_str(&n.to_string()),
Self::F64(n) => f.write_str(&n.to_string()),
Self::String(s) => write!(f, "{}", s),
Self::Array(a) => write!(f, "{:?}", a),
Self::Object(o) => write!(f, "{:?}", o),
}
}
}
impl Index<&str> for Value {
type Output = Self;
fn index(&self, index: &str) -> &Self {
static NULL: Value = Value::Null;
self.get(index).unwrap_or(&NULL)
}
}
impl Default for Value {
fn default() -> Self {
Self::Null
}
}
struct OwnedDeserializer<'de> {
de: Deserializer<'de>,
}
impl<'de> OwnedDeserializer<'de> {
pub fn from_deserializer(de: Deserializer<'de>) -> Self {
Self { de }
}
#[cfg_attr(not(feature = "no-inline"), inline(always))]
pub fn parse(&mut self) -> Result<Value> {
match self.de.next_() {
b'"' => self.de.parse_str_().map(Value::from),
b'n' => Ok(Value::Null),
b't' => Ok(Value::Bool(true)),
b'f' => Ok(Value::Bool(false)),
b'-' => self.de.parse_number_root(true).map(Value::from),
b'0'..=b'9' => self.de.parse_number_root(false).map(Value::from),
b'[' => self.parse_array(),
b'{' => self.parse_map(),
_c => Err(self.de.error(ErrorType::UnexpectedCharacter)),
}
}
#[cfg_attr(not(feature = "no-inline"), inline(always))]
fn parse_value(&mut self) -> Result<Value> {
match self.de.next_() {
b'"' => self.de.parse_str_().map(Value::from),
b'n' => Ok(Value::Null),
b't' => Ok(Value::Bool(true)),
b'f' => Ok(Value::Bool(false)),
b'-' => self.de.parse_number(true).map(Value::from),
b'0'..=b'9' => self.de.parse_number(false).map(Value::from),
b'[' => self.parse_array(),
b'{' => self.parse_map(),
_c => Err(self.de.error(ErrorType::UnexpectedCharacter)),
}
}
#[cfg_attr(not(feature = "no-inline"), inline(always))]
fn parse_array(&mut self) -> Result<Value> {
let es = self.de.count_elements();
if unlikely!(es == 0) {
self.de.skip();
return Ok(Value::Array(Vec::new()));
}
let mut res = Vec::with_capacity(es);
for _i in 0..es {
res.push(stry!(self.parse_value()));
self.de.skip();
}
Ok(Value::Array(res))
}
#[cfg_attr(not(feature = "no-inline"), inline(always))]
fn parse_map(&mut self) -> Result<Value> {
let es = self.de.count_elements();
if unlikely!(es == 0) {
self.de.skip();
return Ok(Value::Object(Object::new()));
}
let mut res = Object::with_capacity(es);
for _ in 0..es {
self.de.skip();
let key = stry!(self.de.parse_str_());
self.de.skip();
res.insert_nocheck(key.into(), stry!(self.parse_value()));
self.de.skip();
}
Ok(Value::Object(res))
}
}
#[cfg(test)]
mod test {
#![allow(clippy::cognitive_complexity)]
use super::*;
#[test]
fn conversions_i64() {
let v = Value::from(i64::max_value());
assert!(v.is_i128());
assert!(v.is_u128());
assert!(v.is_i64());
assert!(v.is_u64());
assert!(!v.is_i32());
assert!(!v.is_u32());
assert!(!v.is_i16());
assert!(!v.is_u16());
assert!(!v.is_i8());
assert!(!v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
let v = Value::from(i64::min_value());
assert!(v.is_i128());
assert!(!v.is_u128());
assert!(v.is_i64());
assert!(!v.is_u64());
assert!(!v.is_i32());
assert!(!v.is_u32());
assert!(!v.is_i16());
assert!(!v.is_u16());
assert!(!v.is_i8());
assert!(!v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
}
#[test]
fn conversions_i32() {
let v = Value::from(i32::max_value());
assert!(v.is_i128());
assert!(v.is_u128());
assert!(v.is_i64());
assert!(v.is_u64());
assert!(v.is_i32());
assert!(v.is_u32());
assert!(!v.is_i16());
assert!(!v.is_u16());
assert!(!v.is_i8());
assert!(!v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
let v = Value::from(i32::min_value());
assert!(v.is_i128());
assert!(!v.is_u128());
assert!(v.is_i64());
assert!(!v.is_u64());
assert!(v.is_i32());
assert!(!v.is_u32());
assert!(!v.is_i16());
assert!(!v.is_u16());
assert!(!v.is_i8());
assert!(!v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
}
#[test]
fn conversions_i16() {
let v = Value::from(i16::max_value());
assert!(v.is_i128());
assert!(v.is_u128());
assert!(v.is_i64());
assert!(v.is_u64());
assert!(v.is_i32());
assert!(v.is_u32());
assert!(v.is_i16());
assert!(v.is_u16());
assert!(!v.is_i8());
assert!(!v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
let v = Value::from(i16::min_value());
assert!(v.is_i128());
assert!(!v.is_u128());
assert!(v.is_i64());
assert!(!v.is_u64());
assert!(v.is_i32());
assert!(!v.is_u32());
assert!(v.is_i16());
assert!(!v.is_u16());
assert!(!v.is_i8());
assert!(!v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
assert!(v.is_f64_castable());
}
#[test]
fn conversions_i8() {
let v = Value::from(i8::max_value());
assert!(v.is_i128());
assert!(v.is_u128());
assert!(v.is_i64());
assert!(v.is_u64());
assert!(v.is_i32());
assert!(v.is_u32());
assert!(v.is_i16());
assert!(v.is_u16());
assert!(v.is_i8());
assert!(v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
let v = Value::from(i8::min_value());
assert!(v.is_i128());
assert!(!v.is_u128());
assert!(v.is_i64());
assert!(!v.is_u64());
assert!(v.is_i32());
assert!(!v.is_u32());
assert!(v.is_i16());
assert!(!v.is_u16());
assert!(v.is_i8());
assert!(!v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
}
#[test]
fn conversions_usize() {
let v = Value::from(usize::min_value() as u64);
assert!(v.is_i128());
assert!(v.is_u128());
assert!(v.is_i64());
assert!(v.is_u64());
assert!(v.is_usize());
assert!(v.is_i32());
assert!(v.is_u32());
assert!(v.is_i16());
assert!(v.is_u16());
assert!(v.is_i8());
assert!(v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
}
#[test]
fn conversions_u64() {
let v = Value::from(u64::min_value());
assert!(v.is_i128());
assert!(v.is_u128());
assert!(v.is_i64());
assert!(v.is_u64());
assert!(v.is_i32());
assert!(v.is_u32());
assert!(v.is_i16());
assert!(v.is_u16());
assert!(v.is_i8());
assert!(v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
}
#[test]
fn conversions_u32() {
let v = Value::from(u32::max_value());
assert!(v.is_i128());
assert!(v.is_u128());
assert!(v.is_i64());
assert!(v.is_u64());
assert!(!v.is_i32());
assert!(v.is_u32());
assert!(!v.is_i16());
assert!(!v.is_u16());
assert!(!v.is_i8());
assert!(!v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
}
#[test]
fn conversions_u16() {
let v = Value::from(u16::max_value());
assert!(v.is_i128());
assert!(v.is_u128());
assert!(v.is_i64());
assert!(v.is_u64());
assert!(v.is_i32());
assert!(v.is_u32());
assert!(!v.is_i16());
assert!(v.is_u16());
assert!(!v.is_i8());
assert!(!v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
}
#[test]
fn conversions_u8() {
let v = Value::from(u8::max_value());
assert!(v.is_i128());
assert!(v.is_u128());
assert!(v.is_i64());
assert!(v.is_u64());
assert!(v.is_i32());
assert!(v.is_u32());
assert!(v.is_i16());
assert!(v.is_u16());
assert!(!v.is_i8());
assert!(v.is_u8());
assert!(!v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
}
#[test]
fn conversions_f64() {
let v = Value::from(std::f64::MAX);
assert!(!v.is_i64());
assert!(!v.is_u64());
assert!(v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
let v = Value::from(std::f64::MIN);
assert!(!v.is_i64());
assert!(!v.is_u64());
assert!(v.is_f64());
assert!(!v.is_f32());
assert!(v.is_f64_castable());
}
#[test]
fn conversions_f32() {
let v = Value::from(std::f32::MAX);
assert!(!v.is_i64());
assert!(!v.is_u64());
assert!(v.is_f64());
assert!(v.is_f32());
assert!(v.is_f64_castable());
let v = Value::from(std::f32::MIN);
assert!(!v.is_i64());
assert!(!v.is_u64());
assert!(v.is_f64());
assert!(v.is_f32());
assert!(v.is_f64_castable());
}
#[test]
fn conversions_array() {
let v = Value::from(vec![true]);
assert!(v.is_array());
assert_eq!(v.value_type(), ValueType::Array);
}
#[test]
fn conversions_bool() {
let v = Value::from(true);
assert!(v.is_bool());
assert_eq!(v.value_type(), ValueType::Bool);
}
#[test]
fn conversions_float() {
let v = Value::from(42.0);
assert!(v.is_f64());
assert_eq!(v.value_type(), ValueType::F64);
}
#[test]
fn conversions_int() {
let v = Value::from(42);
assert!(v.is_i64());
assert_eq!(v.value_type(), ValueType::I64);
}
#[test]
fn conversions_null() {
let v = Value::from(());
assert!(v.is_null());
assert_eq!(v.value_type(), ValueType::Null);
}
#[test]
fn conversions_object() {
let v = Value::Object(Object::new());
assert!(v.is_object());
assert_eq!(v.value_type(), ValueType::Object);
}
#[test]
fn conversions_str() {
let v = Value::from("bla");
assert!(v.is_str());
assert_eq!(v.value_type(), ValueType::String);
}
use proptest::prelude::*;
fn arb_value() -> BoxedStrategy<Value> {
let leaf = prop_oneof![
Just(Value::Null),
any::<bool>().prop_map(Value::Bool),
any::<i64>().prop_map(Value::I64),
any::<f64>().prop_map(Value::F64),
".*".prop_map(Value::from),
];
leaf.prop_recursive(
8,
256,
10,
|inner| {
prop_oneof![
prop::collection::vec(inner.clone(), 0..10).prop_map(Value::Array),
prop::collection::hash_map(".*", inner.clone(), 0..10)
.prop_map(|m| Value::Object(m.into_iter().collect())),
]
},
)
.boxed()
}
proptest! {
#![proptest_config(ProptestConfig {
.. ProptestConfig::default()
})]
#[test]
fn prop_to_owned(owned in arb_value()) {
use crate::BorrowedValue;
let borrowed: BorrowedValue = owned.clone().into();
assert_eq!(owned, borrowed);
}
#[test]
fn prop_serialize_deserialize(owned in arb_value()) {
dbg!(&owned);
let mut string = owned.encode();
dbg!(&string);
let mut bytes = unsafe{ string.as_bytes_mut()};
let decoded = to_value(&mut bytes).expect("Failed to decode");
assert_eq!(owned, decoded)
}
#[test]
fn prop_f64_cmp(f in proptest::num::f64::NORMAL) {
#[allow(clippy::float_cmp)]
let v: Value = f.into();
assert_eq!(v, f)
}
#[test]
fn prop_f32_cmp(f in proptest::num::f32::NORMAL) {
#[allow(clippy::float_cmp)]
let v: Value = f.into();
assert_eq!(v, f)
}
#[test]
fn prop_i64_cmp(f in proptest::num::i64::ANY) {
let v: Value = f.into();
assert_eq!(v, f)
}
#[test]
fn prop_i32_cmp(f in proptest::num::i32::ANY) {
let v: Value = f.into();
assert_eq!(v, f)
}
#[test]
fn prop_i16_cmp(f in proptest::num::i16::ANY) {
let v: Value = f.into();
assert_eq!(v, f)
}
#[test]
fn prop_i8_cmp(f in proptest::num::i8::ANY) {
let v: Value = f.into();
assert_eq!(v, f)
}
#[test]
fn prop_u64_cmp(f in (0_u64..=(i64::max_value() as u64))) {
let v: Value = f.into();
assert_eq!(v, f)
}
#[allow(clippy::cast_possible_truncation)]
#[test]
fn prop_usize_cmp(f in (0_usize..=(i64::max_value() as usize))) {
let v: Value = f.into();
assert_eq!(v, f)
}
#[test]
fn prop_u32_cmp(f in proptest::num::u32::ANY) {
let v: Value = f.into();
assert_eq!(v, f)
}
#[test]
fn prop_u16_cmp(f in proptest::num::u16::ANY) {
let v: Value = f.into();
assert_eq!(v, f)
}
#[test]
fn prop_u8_cmp(f in proptest::num::u8::ANY) {
let v: Value = f.into();
assert_eq!(v, &f);
assert_eq!(v, f);
}
#[test]
fn prop_string_cmp(f in ".*") {
let v: Value = f.clone().into();
assert_eq!(v, f.as_str());
assert_eq!(v, f);
}
}
#[test]
fn test_union_cmp() {
let v: Value = ().into();
assert_eq!(v, ())
}
#[test]
fn test_bool_cmp() {
let v: Value = true.into();
assert_eq!(v, true);
let v: Value = false.into();
assert_eq!(v, false);
}
}