2017-05-04 59 views
2

我有一些函数会在失败时返回不同的错误类型。在不使用map_err的情况下使用带有不同结果错误类型的and_then

首先我有一个生成器,它包含以下方法:

#[derive(Debug)] 
pub enum BuilderError { 
    ElementMissing(&'static str), 
} 

pub fn spawn(self) -> Result<ServiceStatus, BuilderError> 

所以它会在失败时返回一个BuildError。

现在,我有另一个函数将返回另一个错误:

#[derive(Debug)] 
pub enum XmlError { 
    XmlCreationFailed(writer::Error), 
    ConversionToUtf8(FromUtf8Error), 
} 

pub fn create_xml(service_status: super::ServiceStatus) -> Result<String, XmlError> 

的想法是,我用的是Builder创建一个ServiceStatus对象,并用它来创建create_xml功能的XML字符串。

为了做到这一点,我有这样的代码:

#[derive(Debug)] 
pub enum WebserviceError { 
    XmlError(XmlError), 
    BuilderError(BuilderError), 
} 

impl std::error::Error for WebserviceError { 
    ... 
} 

impl From<XmlError> for WebserviceError { 
    fn from(error: XmlError) -> WebserviceError { 
     WebserviceError::XmlError(error) 
    } 
} 

impl From<BuilderError> for WebserviceError { 
    fn from(error: BuilderError) -> WebserviceError { 
     WebserviceError::BuilderError(error) 
    } 
} 

fn test() -> Result<String, status::WebserviceError> { 
    ... 
    let service_status = builder.spawn()?; 
    let xml = status::create_xml(service_status)?; 
    Ok(xml) 
} 

现在,我想我可以做的更好使用and_then而不是使用?运营商:

fn test() -> Result<String, status::WebserviceError> { 
    ... 
    builder 
     .spawn() 
     .map_err(status::WebserviceError::BuilderError) 
     .and_then(|hue| status::create_xml(hue).map_err(status::WebserviceError::XmlError)) 
} 

这种解决方案也是如此,但现在我需要显式调用map_err从BuilderError或XmlError转换为WebserviceError ...

所以,我的问题是,我可以做的更好?我觉得像这样的解决方案将是理想的:

fn test() -> Result<String, status::WebserviceError> { 
    ... 
    builder 
     .spawn() 
     .and_then(status::create_xml) 
} 

感谢

+0

我认为你可以做'.map_err(Into :: into)'让rustc找出哪个impl适用。但我更喜欢用'?'来解决这个问题 – trentcl

回答

3

一些庭审结束后,这里是解决方案:

trait CustomAndThen<T, E> { 
    fn and_then2<U, E2, F: FnOnce(T) -> Result<U, E2>>(self, op: F) -> Result<U, E> 
     where E: std::convert::From<E2>; 
} 

impl<T, E> CustomAndThen<T, E> for Result<T, E> { 
    fn and_then2<U, E2, F: FnOnce(T) -> Result<U, E2>>(self, op: F) -> Result<U, E> 
     where E: std::convert::From<E2> 
    { 
     match self { 
      Ok(t) => op(t).map_err(From::from), 
      Err(e) => Err(e), 
     } 
    } 
} 

... 

Ok(builder) 
    .and_then2(status::ServiceStatusBuilder::spawn) 
    .and_then2(status::create_xml) 

这将为Result类型的定制and_then功能,将使其内的转换,清除代码

相关问题