|
|
@ -40,24 +40,35 @@ where
|
|
|
|
S: Stream<Item = Result<U, E>> + 'a,
|
|
|
|
S: Stream<Item = Result<U, E>> + 'a,
|
|
|
|
{
|
|
|
|
{
|
|
|
|
Box::pin(async move {
|
|
|
|
Box::pin(async move {
|
|
|
|
// Using `scan` here because it is able to stop the stream early
|
|
|
|
// Using `take_while` here because it is able to stop the stream early
|
|
|
|
// if a failure occurs
|
|
|
|
// if a failure occurs
|
|
|
|
|
|
|
|
let mut is_error = false;
|
|
|
|
let mut found_error = None;
|
|
|
|
let mut found_error = None;
|
|
|
|
let out = <T as Product<U>>::product(stream.scan(&mut found_error, |error, elem| {
|
|
|
|
let out = <T as Product<U>>::product(
|
|
|
|
match elem {
|
|
|
|
stream
|
|
|
|
Ok(elem) => Some(elem),
|
|
|
|
.take_while(|elem| {
|
|
|
|
|
|
|
|
// Stop processing the stream on `Err`
|
|
|
|
|
|
|
|
!is_error
|
|
|
|
|
|
|
|
&& (elem.is_ok() || {
|
|
|
|
|
|
|
|
is_error = true;
|
|
|
|
|
|
|
|
// Capture first `Err`
|
|
|
|
|
|
|
|
true
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
.filter_map(|elem| match elem {
|
|
|
|
|
|
|
|
Ok(value) => Some(value),
|
|
|
|
Err(err) => {
|
|
|
|
Err(err) => {
|
|
|
|
**error = Some(err);
|
|
|
|
found_error = Some(err);
|
|
|
|
// Stop processing the stream on error
|
|
|
|
|
|
|
|
None
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}),
|
|
|
|
}))
|
|
|
|
)
|
|
|
|
.await;
|
|
|
|
.await;
|
|
|
|
|
|
|
|
|
|
|
|
match found_error {
|
|
|
|
if is_error {
|
|
|
|
Some(err) => Err(err),
|
|
|
|
Err(found_error.unwrap())
|
|
|
|
None => Ok(out),
|
|
|
|
} else {
|
|
|
|
|
|
|
|
Ok(out)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|