feat: add stream fork/join

This commit is contained in:
2022-03-19 21:06:39 +08:00
parent f1fcb8d6f4
commit 43bd89f477
2 changed files with 103 additions and 84 deletions

View File

@@ -0,0 +1,98 @@
use std::future::Future;
use std::io::{self, ErrorKind};
use std::pin::Pin;
use std::task::{Context, Poll};
use std::time::Duration;
use tokio::io::{AsyncRead, AsyncReadExt, AsyncWrite, AsyncWriteExt, ReadBuf};
use tokio::sync::mpsc::{channel, Receiver, Sender};
use tokio::time::sleep;
#[tokio::main]
async fn main() {
let (sender, receiver) = channel::<StreamMessage>(512);
let stream = Stream { sender, receiver };
let (mut s, mut w) = tokio::io::split(stream);
let t = tokio::spawn(async move {
let mut buf = Vec::<u8>::with_capacity(1024);
loop {
match s.read_buf(&mut buf).await {
Ok(len) => println!("Len: {}, Buf: {:?}", len, &buf[..len]),
Err(e) => {
println!("Err: {:?}", e);
if e.to_string().contains("broken pipe") {
break;
}
}
}
}
});
sleep(Duration::from_secs(1)).await;
let _ = w.write_all(b"hello world").await;
sleep(Duration::from_secs(1)).await;
let _ = w.write_all(b"hello world").await;
sleep(Duration::from_secs(1)).await;
let _ = w.write_all(b"hello world").await;
let a = tokio::join!(w.shutdown());
let b = tokio::join!(t);
println!("{:?}, {:?}", a, b)
}
#[derive(Debug)]
enum StreamMessage {
Data(Vec<u8>),
Flush,
Close,
}
struct Stream {
sender: Sender<StreamMessage>,
receiver: Receiver<StreamMessage>,
}
impl AsyncRead for Stream {
fn poll_read(mut self: Pin<&mut Self>, cx: &mut Context<'_>, buf: &mut ReadBuf<'_>) -> Poll<io::Result<()>> {
let m = self.receiver.poll_recv(cx);
match m {
Poll::Pending => Poll::Pending,
Poll::Ready(m_opt) => match m_opt {
None => Poll::Ready(Err(io::Error::new(ErrorKind::BrokenPipe, "broken pipe"))),
Some(StreamMessage::Close) => {
self.receiver.close();
Poll::Ready(Ok(()))
}
Some(StreamMessage::Flush) => Poll::Ready(Ok(())),
Some(StreamMessage::Data(mm)) => {
buf.put_slice(mm.as_slice());
Poll::Ready(Ok(()))
}
}
}
}
}
impl AsyncWrite for Stream {
fn poll_write(self: Pin<&mut Self>, cx: &mut Context<'_>, buf: &[u8]) -> Poll<io::Result<usize>> {
pool_send(self, cx, StreamMessage::Data(Vec::from(buf)), buf.len())
}
fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
pool_send(self, cx, StreamMessage::Flush, ())
}
fn poll_shutdown(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
pool_send(self, cx, StreamMessage::Close, ())
}
}
fn pool_send<R>(s: Pin<&mut Stream>, cx: &mut Context<'_>, stream_message: StreamMessage, r: R) -> Poll<io::Result<R>> {
let fut = s.sender.send(stream_message);
tokio::pin!( fut);
match fut.poll(cx) {
Poll::Pending => Poll::Pending,
Poll::Ready(Ok(_)) => Poll::Ready(Ok(r)),
Poll::Ready(Err(e)) => Poll::Ready(Err(io::Error::new(ErrorKind::BrokenPipe, e.to_string()))),
}
}

View File

@@ -9,90 +9,11 @@ use tokio::sync::mpsc::{channel, Receiver, Sender};
use tokio::time::sleep;
#[tokio::main]
async fn main() {
let (sender, receiver) = channel::<StreamMessage>(512);
let stream = Stream { sender, receiver };
let (mut s, mut w) = tokio::io::split(stream);
let t = tokio::spawn(async move {
let mut buf = Vec::<u8>::with_capacity(1024);
loop {
match s.read_buf(&mut buf).await {
Ok(len) => println!("Len: {}, Buf: {:?}", len, &buf[..len]),
Err(e) => {
println!("Err: {:?}", e);
if e.to_string().contains("broken pipe") {
break;
}
}
}
}
});
sleep(Duration::from_secs(1)).await;
let _ = w.write_all(b"hello world").await;
sleep(Duration::from_secs(1)).await;
let _ = w.write_all(b"hello world").await;
sleep(Duration::from_secs(1)).await;
let _ = w.write_all(b"hello world").await;
let a = tokio::join!(w.shutdown());
let b = tokio::join!(t);
println!("{:?}, {:?}", a, b)
}
async fn main() {}
#[derive(Debug)]
enum StreamMessage {
Data(Vec<u8>),
Flush,
Close,
}
struct Stream {
sender: Sender<StreamMessage>,
receiver: Receiver<StreamMessage>,
}
impl AsyncRead for Stream {
fn poll_read(mut self: Pin<&mut Self>, cx: &mut Context<'_>, buf: &mut ReadBuf<'_>) -> Poll<io::Result<()>> {
let m = self.receiver.poll_recv(cx);
match m {
Poll::Pending => Poll::Pending,
Poll::Ready(m_opt) => match m_opt {
None => Poll::Ready(Err(io::Error::new(ErrorKind::BrokenPipe, "broken pipe"))),
Some(StreamMessage::Close) => {
self.receiver.close();
Poll::Ready(Ok(()))
}
Some(StreamMessage::Flush) => Poll::Ready(Ok(())),
Some(StreamMessage::Data(mm)) => {
buf.put_slice(mm.as_slice());
Poll::Ready(Ok(()))
}
}
}
}
}
impl AsyncWrite for Stream {
fn poll_write(self: Pin<&mut Self>, cx: &mut Context<'_>, buf: &[u8]) -> Poll<io::Result<usize>> {
pool_send(self, cx, StreamMessage::Data(Vec::from(buf)), buf.len())
}
fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
pool_send(self, cx, StreamMessage::Flush, ())
}
fn poll_shutdown(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
pool_send(self, cx, StreamMessage::Close, ())
}
}
fn pool_send<R>(s: Pin<&mut Stream>, cx: &mut Context<'_>, stream_message: StreamMessage, r: R) -> Poll<io::Result<R>> {
let fut = s.sender.send(stream_message);
tokio::pin!( fut);
match fut.poll(cx) {
Poll::Pending => Poll::Pending,
Poll::Ready(Ok(_)) => Poll::Ready(Ok(r)),
Poll::Ready(Err(e)) => Poll::Ready(Err(io::Error::new(ErrorKind::BrokenPipe, e.to_string()))),
}
enum StreamPackage {
Data(u64, Vec<u8>),
Flush(u64),
Close(u64),
}