2012-08-14 16:17:27 -05:00
|
|
|
/*!
|
|
|
|
|
|
|
|
Higher level communication abstractions.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
// NB: transitionary, de-mode-ing.
|
|
|
|
#[forbid(deprecated_mode)];
|
|
|
|
#[forbid(deprecated_pattern)];
|
|
|
|
|
2012-09-04 13:23:53 -05:00
|
|
|
use pipes::{Channel, Recv, Chan, Port, Selectable};
|
2012-08-14 16:17:27 -05:00
|
|
|
|
|
|
|
export DuplexStream;
|
|
|
|
|
|
|
|
/// An extension of `pipes::stream` that allows both sending and receiving.
|
2012-08-28 13:11:15 -05:00
|
|
|
struct DuplexStream<T: send, U: send> : Channel<T>, Recv<U>, Selectable {
|
|
|
|
priv chan: Chan<T>;
|
|
|
|
priv port: Port <U>;
|
2012-08-14 16:17:27 -05:00
|
|
|
|
|
|
|
fn send(+x: T) {
|
|
|
|
self.chan.send(x)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn try_send(+x: T) -> bool {
|
|
|
|
self.chan.try_send(x)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn recv() -> U {
|
|
|
|
self.port.recv()
|
|
|
|
}
|
|
|
|
|
2012-08-20 14:23:37 -05:00
|
|
|
fn try_recv() -> Option<U> {
|
2012-08-14 16:17:27 -05:00
|
|
|
self.port.try_recv()
|
|
|
|
}
|
|
|
|
|
|
|
|
pure fn peek() -> bool {
|
|
|
|
self.port.peek()
|
|
|
|
}
|
|
|
|
|
2012-08-28 13:11:15 -05:00
|
|
|
pure fn header() -> *pipes::PacketHeader {
|
2012-08-14 16:17:27 -05:00
|
|
|
self.port.header()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a bidirectional stream.
|
|
|
|
fn DuplexStream<T: send, U: send>()
|
|
|
|
-> (DuplexStream<T, U>, DuplexStream<U, T>)
|
|
|
|
{
|
|
|
|
let (c2, p1) = pipes::stream();
|
|
|
|
let (c1, p2) = pipes::stream();
|
|
|
|
(DuplexStream {
|
|
|
|
chan: c1,
|
|
|
|
port: p1
|
|
|
|
},
|
|
|
|
DuplexStream {
|
|
|
|
chan: c2,
|
|
|
|
port: p2
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
#[test]
|
|
|
|
fn DuplexStream1() {
|
|
|
|
let (left, right) = DuplexStream();
|
|
|
|
|
|
|
|
left.send(~"abc");
|
|
|
|
right.send(123);
|
|
|
|
|
|
|
|
assert left.recv() == 123;
|
|
|
|
assert right.recv() == ~"abc";
|
|
|
|
}
|
|
|
|
}
|