use quick_xml::events::{BytesStart, Event}; use quick_xml::name::ResolveResult; use std::io::BufRead; pub use xml_derive::XmlDeserialize; pub use xml_derive::XmlDocument; use crate::XmlError; use crate::XmlRootTag; pub trait XmlDeserialize: Sized { fn deserialize( reader: &mut quick_xml::NsReader, start: &BytesStart, empty: bool, ) -> Result; } pub trait XmlDocument: XmlDeserialize { fn parse(reader: quick_xml::NsReader) -> Result; #[inline] fn parse_reader(input: R) -> Result where Self: XmlDeserialize, { let mut reader = quick_xml::NsReader::from_reader(input); reader.config_mut().trim_text(true); Self::parse(reader) } #[inline] fn parse_str(s: &str) -> Result { Self::parse_reader(s.as_bytes()) } } impl XmlDocument for T { fn parse(mut reader: quick_xml::NsReader) -> Result where Self: XmlDeserialize, { let mut buf = Vec::new(); loop { let event = reader.read_event_into(&mut buf)?; let empty = matches!(event, Event::Empty(_)); match event { Event::Decl(_) => { /* ignore this */ } Event::Comment(_) => { /* ignore this */ } Event::Start(start) | Event::Empty(start) => { let (ns, name) = reader.resolve_element(start.name()); let matches = match (Self::root_ns(), &ns, name) { // Wrong tag (_, _, name) if name.as_ref() != Self::root_tag() => false, // Wrong namespace (Some(root_ns), ns, _) if &ResolveResult::Bound(root_ns) != ns => false, _ => true, }; if !matches { let root_ns = Self::root_ns(); return Err(XmlError::InvalidTag( format!("{ns:?}"), String::from_utf8_lossy(name.as_ref()).to_string(), format!("{root_ns:?}"), String::from_utf8_lossy(Self::root_tag()).to_string(), )); }; return Self::deserialize(&mut reader, &start, empty); } _ => return Err(XmlError::UnsupportedEvent("unknown, todo")), }; } } } impl XmlDeserialize for () { fn deserialize( reader: &mut quick_xml::NsReader, start: &BytesStart, empty: bool, ) -> Result { if empty { return Ok(()); } let mut buf = Vec::new(); loop { match reader.read_event_into(&mut buf)? { Event::End(e) if e.name() == start.name() => return Ok(()), Event::Eof => return Err(XmlError::Eof), _ => {} }; } } }