use alloc::boxed::Box;
use crate::error::Error;
use crate::rxml_util::Item;
use crate::AsXml;
use core::fmt;
use bytes::BytesMut;
pub struct OptionAsXml<T: Iterator>(Option<T>);
impl<T: Iterator> OptionAsXml<T> {
pub fn new(inner: Option<T>) -> Self {
Self(inner)
}
}
impl<'x, T: Iterator<Item = Result<Item<'x>, Error>>> Iterator for OptionAsXml<T> {
type Item = Result<Item<'x>, Error>;
fn next(&mut self) -> Option<Self::Item> {
self.0.as_mut()?.next()
}
}
impl<T: AsXml> AsXml for Option<T> {
type ItemIter<'x>
= OptionAsXml<T::ItemIter<'x>>
where
T: 'x;
fn as_xml_iter(&self) -> Result<Self::ItemIter<'_>, Error> {
match self {
Some(ref value) => Ok(OptionAsXml(Some(T::as_xml_iter(value)?))),
None => Ok(OptionAsXml(None)),
}
}
}
pub struct BoxAsXml<T: Iterator>(Box<T>);
impl<'x, T: Iterator<Item = Result<Item<'x>, Error>>> Iterator for BoxAsXml<T> {
type Item = Result<Item<'x>, Error>;
fn next(&mut self) -> Option<Self::Item> {
self.0.next()
}
}
impl<T: AsXml> AsXml for Box<T> {
type ItemIter<'x>
= BoxAsXml<T::ItemIter<'x>>
where
T: 'x;
fn as_xml_iter(&self) -> Result<Self::ItemIter<'_>, Error> {
Ok(BoxAsXml(Box::new(T::as_xml_iter(self)?)))
}
}
impl<T: AsXml, E> AsXml for Result<T, E>
where
for<'a> Error: From<&'a E>,
{
type ItemIter<'x>
= T::ItemIter<'x>
where
Self: 'x;
fn as_xml_iter(&self) -> Result<Self::ItemIter<'_>, Error> {
match self {
Self::Ok(v) => Ok(v.as_xml_iter()?),
Self::Err(e) => Err(e.into()),
}
}
}
pub struct PrintRawXml<'x, T>(pub &'x T);
impl<'x, T: AsXml> fmt::Display for PrintRawXml<'x, T> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let iter = match self.0.as_xml_iter() {
Ok(iter) => iter,
Err(err) => return write!(f, "<failed to serialize PrintRawXml: {:?}>", err),
};
let mut writer = rxml::writer::Encoder::new();
let mut buf = BytesMut::new();
for item in iter {
let item = match item {
Ok(item) => item,
Err(err) => return write!(f, "<failed to serialize PrintRawXml: {:?}>", err),
};
if let Err(err) = writer.encode(item.as_rxml_item(), &mut buf) {
return write!(f, "<failed to serialize PrintRawXml: {:?}>", err);
}
}
write!(f, "{}", std::str::from_utf8(&buf).unwrap())
}
}
#[cfg(test)]
mod tests {
use super::*;
use alloc::{borrow::Cow, vec};
#[test]
fn option_as_xml_terminates_immediately_for_none() {
let mut iter = OptionAsXml::<core::iter::Empty<_>>(None);
match iter.next() {
None => (),
other => panic!("unexpected item: {:?}", other),
}
}
#[test]
fn option_as_xml_passes_values_from_inner_some() {
let inner = vec![
Ok(Item::Text(Cow::Borrowed("hello world"))),
Ok(Item::ElementFoot),
];
let mut iter = OptionAsXml(Some(inner.into_iter()));
match iter.next() {
Some(Ok(Item::Text(text))) => {
assert_eq!(text, "hello world");
}
other => panic!("unexpected item: {:?}", other),
}
match iter.next() {
Some(Ok(Item::ElementFoot)) => (),
other => panic!("unexpected item: {:?}", other),
}
match iter.next() {
None => (),
other => panic!("unexpected item: {:?}", other),
}
}
#[test]
fn box_as_xml_passes_values_from_inner() {
let inner = vec![
Ok(Item::Text(Cow::Borrowed("hello world"))),
Ok(Item::ElementFoot),
];
let mut iter = BoxAsXml(Box::new(inner.into_iter()));
match iter.next() {
Some(Ok(Item::Text(text))) => {
assert_eq!(text, "hello world");
}
other => panic!("unexpected item: {:?}", other),
}
match iter.next() {
Some(Ok(Item::ElementFoot)) => (),
other => panic!("unexpected item: {:?}", other),
}
match iter.next() {
None => (),
other => panic!("unexpected item: {:?}", other),
}
}
}