use super::Object; pub enum Node { Array(Vec>), Object(T), Link(Box), Empty, } impl From> for Node { fn from(value: Option) -> Self { match value { Some(x) => Node::Object(x), None => Node::Empty, } } } impl Node { pub fn get(&self) -> Option<&T> { match self { Node::Empty | Node::Link(_) => None, Node::Object(x) => Some(x), Node::Array(v) => match v.iter().find_map(|x| match x { Node::Object(x) => Some(x), _ => None, }) { Some(x) => Some(x), None => None, }, } } pub fn all(&self) -> Option> { match self { Node::Empty | Node::Link(_) => None, Node::Object(x) => Some(vec![x]), Node::Array(v) => Some(v.iter().filter_map(|x| match x { Node::Object(x) => Some(x), _ => None, }).collect()), } } pub fn is_empty(&self) -> bool { match self { Node::Empty | Node::Link(_) => true, Node::Object(_) | Node::Array(_) => false, } } pub fn len(&self) -> usize { match self { Node::Empty => 0, Node::Link(_) => 0, Node::Object(_) => 1, Node::Array(v) => v.len(), } } } impl Node where T : Object { pub fn id(&self) -> Option<&str> { match self { Node::Empty => None, Node::Link(uri) => Some(uri.href()), Node::Object(obj) => obj.id(), Node::Array(arr) => arr.first()?.id(), } } } impl From> for Node { fn from(value: Option<&str>) -> Self { match value { Some(x) => Node::Link(Box::new(x.to_string())), None => Node::Empty, } } } impl From<&str> for Node { fn from(value: &str) -> Self { Node::Link(Box::new(value.to_string())) } } impl From for Node { fn from(value: serde_json::Value) -> Self { match value { serde_json::Value::String(uri) => Node::Link(Box::new(uri)), serde_json::Value::Object(_) => match value.get("href") { Some(_) => Node::Object(value), None => Node::Link(Box::new(value)), }, serde_json::Value::Array(arr) => Node::Array( arr .into_iter() .map(Self::from) .collect() ), _ => Node::Empty, } } } impl Node{ pub async fn fetch(&mut self) -> reqwest::Result<()> { if let Node::Link(link) = self { *self = reqwest::get(link.href()) .await? .json::() .await? .into(); } Ok(()) } } pub(crate) trait NodeExtractor { fn node(&self, id: &str) -> Node; fn node_vec(&self, id: &str) -> Node; } impl NodeExtractor for serde_json::Value { fn node(&self, id: &str) -> Node { match self.get(id) { None => Node::Empty, Some(x) => Node::from(x.clone()), } } fn node_vec(&self, id: &str) -> Node { match self.get(id) { None => Node::Empty, Some(x) => Node::from(x.clone()), } } } pub(crate) trait InsertStr { fn insert_str(&mut self, k: &str, v: Option<&str>); fn insert_timestr(&mut self, k: &str, t: Option>); } impl InsertStr for serde_json::Map { fn insert_str(&mut self, k: &str, v: Option<&str>) { if let Some(v) = v { self.insert( k.to_string(), serde_json::Value::String(v.to_string()), ); } } fn insert_timestr(&mut self, k: &str, t: Option>) { if let Some(published) = t { self.insert( k.to_string(), serde_json::Value::String(published.to_rfc3339()), ); } } }