2019-10-20 14:05:55 +00:00
|
|
|
use std::pin::Pin;
|
|
|
|
|
|
|
|
use crate::prelude::*;
|
|
|
|
use crate::stream::{Stream, Sum};
|
|
|
|
|
|
|
|
impl<T, U, E> Sum<Result<U, E>> for Result<T, E>
|
|
|
|
where
|
|
|
|
T: Sum<U>,
|
|
|
|
{
|
|
|
|
#[doc = r#"
|
|
|
|
Takes each element in the `Stream`: if it is an `Err`, no further
|
|
|
|
elements are taken, and the `Err` is returned. Should no `Err` occur,
|
|
|
|
the sum of all elements is returned.
|
|
|
|
|
|
|
|
# Examples
|
|
|
|
|
|
|
|
This sums up every integer in a vector, rejecting the sum if a negative
|
|
|
|
element is encountered:
|
|
|
|
|
|
|
|
```
|
|
|
|
# fn main() { async_std::task::block_on(async {
|
|
|
|
#
|
|
|
|
use async_std::prelude::*;
|
2019-11-03 12:40:51 +00:00
|
|
|
use async_std::stream;
|
2019-10-20 14:05:55 +00:00
|
|
|
|
2019-11-03 12:40:51 +00:00
|
|
|
let v = stream::from_iter(vec![1, 2]);
|
2019-10-20 14:05:55 +00:00
|
|
|
let res: Result<i32, &'static str> = v.map(|x|
|
|
|
|
if x < 0 {
|
|
|
|
Err("Negative element found")
|
|
|
|
} else {
|
|
|
|
Ok(x)
|
|
|
|
}).sum().await;
|
|
|
|
assert_eq!(res, Ok(3));
|
|
|
|
#
|
|
|
|
# }) }
|
|
|
|
```
|
|
|
|
"#]
|
|
|
|
fn sum<'a, S>(stream: S) -> Pin<Box<dyn Future<Output = Result<T, E>> + 'a>>
|
|
|
|
where S: Stream<Item = Result<U, E>> + 'a
|
|
|
|
{
|
|
|
|
Box::pin(async move {
|
|
|
|
// Using `scan` here because it is able to stop the stream early
|
|
|
|
// if a failure occurs
|
|
|
|
let mut found_error = None;
|
|
|
|
let out = <T as Sum<U>>::sum(stream
|
|
|
|
.scan((), |_, elem| {
|
|
|
|
match elem {
|
|
|
|
Ok(elem) => Some(elem),
|
|
|
|
Err(err) => {
|
|
|
|
found_error = Some(err);
|
|
|
|
// Stop processing the stream on error
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})).await;
|
|
|
|
match found_error {
|
|
|
|
Some(err) => Err(err),
|
|
|
|
None => Ok(out)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|