1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77
//! Operators that separate one stream into two streams based on some condition
use crate::container::{Container, SizableContainer, PushInto};
use crate::dataflow::channels::pact::Pipeline;
use crate::dataflow::operators::generic::builder_rc::OperatorBuilder;
use crate::dataflow::{Scope, StreamCore};
/// Extension trait for `Stream`.
pub trait OkErr<S: Scope, C: Container> {
/// Takes one input stream and splits it into two output streams.
/// For each record, the supplied closure is called with the data.
/// If it returns `Ok(x)`, then `x` will be sent
/// to the first returned stream; otherwise, if it returns `Err(e)`,
/// then `e` will be sent to the second.
///
/// # Examples
/// ```
/// use timely::dataflow::operators::ToStream;
/// use timely::dataflow::operators::core::{OkErr, Inspect};
///
/// timely::example(|scope| {
/// let (odd, even) = (0..10)
/// .to_stream(scope)
/// .ok_err(|x| if x % 2 == 0 { Ok(x) } else { Err(x) });
///
/// even.container::<Vec<_>>().inspect(|x| println!("even: {:?}", x));
/// odd.container::<Vec<_>>().inspect(|x| println!("odd: {:?}", x));
/// });
/// ```
fn ok_err<C1, D1, C2, D2, L>(
&self,
logic: L,
) -> (StreamCore<S, C1>, StreamCore<S, C2>)
where
C1: SizableContainer + PushInto<D1>,
C2: SizableContainer + PushInto<D2>,
L: FnMut(C::Item<'_>) -> Result<D1,D2>+'static
;
}
impl<S: Scope, C: Container> OkErr<S, C> for StreamCore<S, C> {
fn ok_err<C1, D1, C2, D2, L>(
&self,
mut logic: L,
) -> (StreamCore<S, C1>, StreamCore<S, C2>)
where
C1: SizableContainer + PushInto<D1>,
C2: SizableContainer + PushInto<D2>,
L: FnMut(C::Item<'_>) -> Result<D1,D2>+'static
{
let mut builder = OperatorBuilder::new("OkErr".to_owned(), self.scope());
let mut input = builder.new_input(self, Pipeline);
let (mut output1, stream1) = builder.new_output();
let (mut output2, stream2) = builder.new_output();
builder.build(move |_| {
move |_frontiers| {
let mut output1_handle = output1.activate();
let mut output2_handle = output2.activate();
input.for_each(|time, data| {
let mut out1 = output1_handle.session(&time);
let mut out2 = output2_handle.session(&time);
for datum in data.drain() {
match logic(datum) {
Ok(datum) => out1.give(datum),
Err(datum) => out2.give(datum),
}
}
});
}
});
(stream1, stream2)
}
}