rbdc 4.9.10

The Rust SQL Toolkit and ORM Library. An async, pure Rust SQL crate featuring compile-time Dynamic SQL
Documentation
use rbs::Value;
use std::fmt::{Debug, Display, Formatter};
use std::str::FromStr;

use serde::de::Error;
use serde::{Deserialize, Deserializer, Serialize, Serializer};

/// support deserialize JsonStruct or json string
/// for example:
/// ```rust
/// #[derive(Clone, Debug, serde::Serialize, serde::Deserialize)]
/// pub struct Account {
///     pub id: Option<String>,
///     pub name: Option<String>,
/// }
///
///// #[derive(Clone, serde::Serialize, serde::Deserialize)]
///// pub struct User {
/////     pub id: Option<String>,
/////     //Deserialize json column.
/////     //maybe account is json_str: `{"id":"1","name":"v"}`  or json: {"id":"1","name":"v"}
/////     #[serde(deserialize_with = "rbatis::rbdc::types::deserialize_maybe_str")]
/////     pub account: Account,
///// }
///
/// ```
pub fn deserialize_maybe_str<'de, D, T>(deserializer: D) -> Result<T, D::Error>
where
    D: Deserializer<'de>,
    T: serde::de::DeserializeOwned,
{
    use serde::de::Error;
    let account_value: serde_json::Value = Deserialize::deserialize(deserializer)?;
    match account_value {
        serde_json::Value::String(account_str) => {
            let typename = std::any::type_name::<T>();
            if typename == std::any::type_name::<String>()
                || typename == std::any::type_name::<Option<String>>()
            {
                let account_json: T =
                    serde_json::from_value(serde_json::Value::String(account_str))
                        .map_err(D::Error::custom)?;
                Ok(account_json)
            } else {
                let account_json: T =
                    serde_json::from_str(&account_str).map_err(D::Error::custom)?;
                Ok(account_json)
            }
        }
        _ => {
            let account_json: T =
                serde_json::from_str(&account_value.to_string()).map_err(D::Error::custom)?;
            Ok(account_json)
        }
    }
}

#[derive(serde::Serialize, Clone, Eq, PartialEq, Hash)]
#[serde(rename = "Json")]
pub struct Json(pub String);

impl<'de> serde::Deserialize<'de> for Json {
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
    where
        D: Deserializer<'de>,
    {
        Ok(Json(
            serde_json::Value::deserialize(deserializer)
                .map_err(|e| Error::custom(e))?
                .to_string(),
        ))
    }
}

impl Default for Json {
    fn default() -> Self {
        Self {
            0: "null".to_string(),
        }
    }
}

impl From<serde_json::Value> for Json {
    fn from(arg: serde_json::Value) -> Self {
        Json(arg.to_string())
    }
}

impl From<Value> for Json {
    fn from(v: Value) -> Self {
        match v {
            Value::Null => Json(v.to_string()),
            Value::Bool(v) => Json(v.to_string()),
            Value::I32(v) => Json(v.to_string()),
            Value::I64(v) => Json(v.to_string()),
            Value::U32(v) => Json(v.to_string()),
            Value::U64(v) => Json(v.to_string()),
            Value::F32(v) => Json(v.to_string()),
            Value::F64(v) => Json(v.to_string()),
            Value::String(mut v) => {
                if (v.starts_with("{") && v.ends_with("}"))
                    || (v.starts_with("[") && v.ends_with("]"))
                    || (v.starts_with("\"") && v.ends_with("\""))
                {
                    //is json-string
                    Json(v)
                } else {
                    v.insert(0, '"');
                    v.push('"');
                    Json(v)
                }
            }
            Value::Binary(v) => Json(String::from_utf8_lossy(&v).into_owned()),
            Value::Array(_) => Json(v.to_string()),
            Value::Map(v) => Json(v.to_string()),
            Value::Ext(_name, v) => Json::from(*v),
        }
    }
}

impl Display for Json {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        write!(f, "{}", self.0)
    }
}

impl Debug for Json {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        write!(f, "Json({})", self.0)
    }
}

impl From<Json> for Value {
    fn from(arg: Json) -> Self {
        Value::Ext("Json", Box::new(Value::String(arg.0)))
    }
}

impl From<Json> for serde_json::Value {
    fn from(arg: Json) -> Self {
        let v: serde_json::Value = serde_json::from_str(&arg.0).unwrap_or_default();
        v
    }
}

impl FromStr for Json {
    type Err = crate::Error;

    fn from_str(s: &str) -> Result<Self, Self::Err> {
        Ok(Self(s.to_string()))
    }
}

/// unstable
/// use:
/// #[derive(Clone, serde::Serialize, serde::Deserialize)]
/// pub struct BizUser {
///     pub id: Option<String>,
///     pub account: Option<JsonV<Account>>,
/// }
#[derive(Clone, Eq, PartialEq, Hash, Debug)]
pub struct JsonV<T: Serialize + serde::de::DeserializeOwned>(pub T);

impl<T: Serialize + serde::de::DeserializeOwned> Serialize for JsonV<T> {
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: Serializer,
    {
        use serde::ser::Error;
        if std::any::type_name::<S::Error>() == std::any::type_name::<rbs::Error>() {
            Json(serde_json::to_string(&self.0).map_err(|e| Error::custom(e.to_string()))?)
                .serialize(serializer)
        } else {
            self.0.serialize(serializer)
        }
    }
}

impl<'de, T: Serialize + serde::de::DeserializeOwned> Deserialize<'de> for JsonV<T> {
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
    where
        D: Deserializer<'de>,
    {
        use serde::de::Error;
        if std::any::type_name::<D::Error>() == std::any::type_name::<rbs::Error>() {
            let mut v = Value::deserialize(deserializer)?;
            let js;
            if let Value::Ext(_ty, buf) = v {
                v = *buf;
            }
            if let Value::Binary(buf) = v {
                js = String::from_utf8(buf).map_err(|e| D::Error::custom(e.to_string()))?;
            } else if let Value::String(buf) = v {
                js = buf;
            } else {
                js = v.to_string();
            }
            if std::any::type_name::<D::Error>() == std::any::type_name::<rbs::Error>() {
                Ok(JsonV(
                    serde_json::from_str(&js).map_err(|e| Error::custom(e.to_string()))?,
                ))
            } else {
                Ok(JsonV(
                    serde_json::from_str(&js).map_err(|e| Error::custom(e.to_string()))?,
                ))
            }
        } else {
            Ok(JsonV(T::deserialize(deserializer)?))
        }
    }
}

impl<T: Serialize + serde::de::DeserializeOwned + Display> Display for JsonV<T> {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        Display::fmt(&self.0, f)
    }
}

#[cfg(test)]
mod test {
    use crate::JsonV;
    use crate::json::Json;
    use rbs::value::map::ValueMap;

    #[test]
    fn test_decode_js_string() {
        let m = rbs::Value::String(r#""aa""#.to_string());
        println!("{}", m);
        assert_eq!(r#""aa""#, Json::from(m).0);
    }

    #[test]
    fn test_decode_js_string_map() {
        let mut m = ValueMap::new();
        m.insert("a".into(), "1".into());
        let m = rbs::Value::Map(m);
        println!("{}", m.to_string());
        assert_eq!(r#"{"a":"1"}"#, Json::from(m).0);
    }

    #[test]
    fn test_decode_js_int_map() {
        let mut m = ValueMap::new();
        m.insert("a".into(), 1.into());
        let m = rbs::Value::Map(m);
        println!("{}", m.to_string());
        assert_eq!(r#"{"a":1}"#, Json::from(m).0);
    }

    #[test]
    fn test_decode_js_int_arr() {
        let arr = rbs::Value::Array(vec![rbs::Value::I64(1), rbs::Value::I64(2)]);
        println!("{}", arr.to_string());
        assert_eq!(r#"[1,2]"#, Json::from(arr).0);
    }

    #[test]
    fn test_decode_js_string_arr() {
        let arr = rbs::Value::Array(vec![
            rbs::Value::String(1.to_string()),
            rbs::Value::String(2.to_string()),
        ]);
        println!("{}", arr.to_string());
        assert_eq!(r#"["1","2"]"#, Json::from(arr).0);
    }

    #[test]
    fn test_decode_raw() {
        let m: Json = serde_json::from_str(r#"{"a":1}"#).unwrap();
        assert_eq!(r#"{"a":1}"#, m.0);
    }

    #[test]
    fn test_encode_jsonv() {
        let source: JsonV<String> = JsonV(1.to_string());
        let v = rbs::value!(source);
        let data = *v.as_ext().unwrap().1.clone();
        assert_eq!(data.into_string().unwrap_or_default(), "\"1\"");
    }
}