use crate::{
rpc::{typed_data::Data, TypedData},
FromVec,
};
use serde::{Deserialize, Serialize};
use serde_json::{to_string, to_value, Value};
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct SignalRMessage {
pub user_id: Option<String>,
pub group_name: Option<String>,
pub target: String,
pub arguments: Vec<Value>,
}
#[doc(hidden)]
impl Into<TypedData> for SignalRMessage {
fn into(self) -> TypedData {
TypedData {
data: Some(Data::Json(
to_string(&self).expect("failed to convert SignalR message to JSON string"),
)),
}
}
}
#[doc(hidden)]
impl FromVec<SignalRMessage> for TypedData {
fn from_vec(vec: Vec<SignalRMessage>) -> Self {
TypedData {
data: Some(Data::Json(
Value::Array(vec.into_iter().map(|m| to_value(m).unwrap()).collect()).to_string(),
)),
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use serde_json::to_value;
#[test]
fn it_serializes_to_json() {
let json = to_string(&SignalRMessage {
user_id: Some("foo".to_owned()),
group_name: Some("bar".to_owned()),
target: "baz".to_owned(),
arguments: vec![
to_value(1).unwrap(),
to_value("foo").unwrap(),
to_value(false).unwrap(),
],
})
.unwrap();
assert_eq!(
json,
r#"{"userId":"foo","groupName":"bar","target":"baz","arguments":[1,"foo",false]}"#
);
}
#[test]
fn it_converts_to_typed_data() {
let message = SignalRMessage {
user_id: Some("foo".to_owned()),
group_name: Some("bar".to_owned()),
target: "baz".to_owned(),
arguments: vec![
to_value(1).unwrap(),
to_value("foo").unwrap(),
to_value(false).unwrap(),
],
};
let data: TypedData = message.into();
assert_eq!(
data.data,
Some(Data::Json(
r#"{"userId":"foo","groupName":"bar","target":"baz","arguments":[1,"foo",false]}"#
.to_string()
))
);
}
}