use alloc::boxed::Box;
use crate::error::Error;
use crate::rxml_util::Item;
use crate::AsXml;
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()),
}
}
}
#[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),
}
}
}