use indexmap::IndexMap;
use serde::{Deserialize, Serialize};
use std::fmt;
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(untagged)]
pub enum Value {
Null,
Bool(bool),
Number(f64),
String(String),
Array(Vec<Value>),
Object(IndexMap<String, Value>),
#[serde(serialize_with = "serialize_undefined")]
Undefined,
}
fn serialize_undefined<S: serde::Serializer>(serializer: S) -> Result<S::Ok, S::Error> {
serializer.serialize_none()
}
impl Value {
pub fn is_truthy(&self) -> bool {
match self {
Value::Null | Value::Undefined => false,
Value::Bool(b) => *b,
Value::Number(n) => *n != 0.0,
Value::String(s) => !s.is_empty(),
Value::Array(_) => true,
Value::Object(_) => true,
}
}
pub fn type_of(&self) -> &'static str {
match self {
Value::Null => "null",
Value::Undefined => "undefined",
Value::Bool(_) => "boolean",
Value::Number(_) => "number",
Value::String(_) => "string",
Value::Array(_) => "array",
Value::Object(_) => "object",
}
}
pub fn is_null(&self) -> bool {
matches!(self, Value::Null | Value::Undefined)
}
pub fn as_bool(&self) -> Option<bool> {
match self {
Value::Bool(b) => Some(*b),
_ => None,
}
}
pub fn as_number(&self) -> Option<f64> {
match self {
Value::Number(n) => Some(*n),
_ => None,
}
}
pub fn as_str(&self) -> Option<&str> {
match self {
Value::String(s) => Some(s),
_ => None,
}
}
pub fn as_string(&self) -> Option<&String> {
match self {
Value::String(s) => Some(s),
_ => None,
}
}
pub fn as_array(&self) -> Option<&Vec<Value>> {
match self {
Value::Array(a) => Some(a),
_ => None,
}
}
pub fn as_array_mut(&mut self) -> Option<&mut Vec<Value>> {
match self {
Value::Array(a) => Some(a),
_ => None,
}
}
pub fn as_object(&self) -> Option<&IndexMap<String, Value>> {
match self {
Value::Object(o) => Some(o),
_ => None,
}
}
pub fn as_object_mut(&mut self) -> Option<&mut IndexMap<String, Value>> {
match self {
Value::Object(o) => Some(o),
_ => None,
}
}
pub fn to_number(&self) -> f64 {
match self {
Value::Null => 0.0,
Value::Undefined => f64::NAN,
Value::Bool(b) => {
if *b {
1.0
} else {
0.0
}
}
Value::Number(n) => *n,
Value::String(s) => s.parse::<f64>().unwrap_or(f64::NAN),
Value::Array(_) => f64::NAN,
Value::Object(_) => f64::NAN,
}
}
pub fn to_display_string(&self) -> String {
match self {
Value::Null => "null".to_string(),
Value::Undefined => "undefined".to_string(),
Value::Bool(b) => b.to_string(),
Value::Number(n) => format_number(*n),
Value::String(s) => s.clone(),
Value::Array(_) | Value::Object(_) => self.to_json_string(),
}
}
pub fn to_json_string(&self) -> String {
match self {
Value::Null | Value::Undefined => "null".to_string(),
Value::Bool(b) => b.to_string(),
Value::Number(n) => format_number(*n),
Value::String(s) => format!("\"{}\"", s.replace('\\', "\\\\").replace('"', "\\\"")),
Value::Array(a) => {
let items: Vec<String> = a.iter().map(|v| v.to_json_string()).collect();
format!("[{}]", items.join(","))
}
Value::Object(o) => {
let pairs: Vec<String> = o
.iter()
.map(|(k, v)| format!("\"{}\":{}", k, v.to_json_string()))
.collect();
format!("{{{}}}", pairs.join(","))
}
}
}
pub fn deep_eq(&self, other: &Value) -> bool {
match (self, other) {
(Value::Null | Value::Undefined, Value::Null | Value::Undefined) => true,
(Value::Bool(a), Value::Bool(b)) => a == b,
(Value::Number(a), Value::Number(b)) => {
if a.is_nan() && b.is_nan() {
return true;
}
(a - b).abs() < f64::EPSILON
}
(Value::String(a), Value::String(b)) => a == b,
(Value::Number(n), Value::String(s)) | (Value::String(s), Value::Number(n)) => {
if let Ok(parsed) = s.parse::<f64>() {
(*n - parsed).abs() < f64::EPSILON
} else {
false
}
}
(Value::Bool(b), Value::Number(n)) | (Value::Number(n), Value::Bool(b)) => {
let bval = if *b { 1.0 } else { 0.0 };
(*n - bval).abs() < f64::EPSILON
}
(Value::Array(a), Value::Array(b)) => {
if a.len() != b.len() {
return false;
}
a.iter().zip(b.iter()).all(|(x, y)| x.deep_eq(y))
}
(Value::Object(a), Value::Object(b)) => {
if a.len() != b.len() {
return false;
}
a.iter()
.all(|(k, v)| b.get(k).is_some_and(|bv| v.deep_eq(bv)))
}
_ => false,
}
}
}
impl fmt::Display for Value {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.to_display_string())
}
}
impl PartialEq for Value {
fn eq(&self, other: &Self) -> bool {
self.deep_eq(other)
}
}
impl Default for Value {
fn default() -> Self {
Value::Null
}
}
impl From<bool> for Value {
fn from(b: bool) -> Self {
Value::Bool(b)
}
}
impl From<f64> for Value {
fn from(n: f64) -> Self {
Value::Number(n)
}
}
impl From<i64> for Value {
fn from(n: i64) -> Self {
Value::Number(n as f64)
}
}
impl From<String> for Value {
fn from(s: String) -> Self {
Value::String(s)
}
}
impl From<&str> for Value {
fn from(s: &str) -> Self {
Value::String(s.to_string())
}
}
impl<T: Into<Value>> From<Vec<T>> for Value {
fn from(v: Vec<T>) -> Self {
Value::Array(v.into_iter().map(Into::into).collect())
}
}
impl From<IndexMap<String, Value>> for Value {
fn from(m: IndexMap<String, Value>) -> Self {
Value::Object(m)
}
}
impl From<serde_json::Value> for Value {
fn from(v: serde_json::Value) -> Self {
match v {
serde_json::Value::Null => Value::Null,
serde_json::Value::Bool(b) => Value::Bool(b),
serde_json::Value::Number(n) => Value::Number(n.as_f64().unwrap_or(0.0)),
serde_json::Value::String(s) => Value::String(s),
serde_json::Value::Array(a) => {
Value::Array(a.into_iter().map(Value::from).collect())
}
serde_json::Value::Object(o) => {
Value::Object(o.into_iter().map(|(k, v)| (k, Value::from(v))).collect())
}
}
}
}
impl From<Value> for serde_json::Value {
fn from(v: Value) -> Self {
match v {
Value::Null | Value::Undefined => serde_json::Value::Null,
Value::Bool(b) => serde_json::Value::Bool(b),
Value::Number(n) => {
serde_json::Number::from_f64(n)
.map(serde_json::Value::Number)
.unwrap_or(serde_json::Value::Null)
}
Value::String(s) => serde_json::Value::String(s),
Value::Array(a) => {
serde_json::Value::Array(a.into_iter().map(serde_json::Value::from).collect())
}
Value::Object(o) => {
serde_json::Value::Object(
o.into_iter()
.map(|(k, v)| (k, serde_json::Value::from(v)))
.collect(),
)
}
}
}
}
pub fn format_number(n: f64) -> String {
if n.is_infinite() {
if n.is_sign_positive() {
"Infinity".to_string()
} else {
"-Infinity".to_string()
}
} else if n.is_nan() {
"NaN".to_string()
} else if n == n.trunc() && n.abs() < 1e15 {
format!("{}", n as i64)
} else {
format!("{}", n)
}
}