forked from Deuxfleurs/garage
153 lines
4.3 KiB
Rust
153 lines
4.3 KiB
Rust
use std::collections::HashMap;
|
|
use std::sync::Arc;
|
|
|
|
use async_trait::async_trait;
|
|
use bytes::Bytes;
|
|
use log::*;
|
|
|
|
use futures::AsyncReadExt;
|
|
use tokio::sync::mpsc;
|
|
|
|
use crate::error::*;
|
|
use crate::send::*;
|
|
use crate::stream::*;
|
|
|
|
/// Structure to warn when the sender is dropped before end of stream was reached, like when
|
|
/// connection to some remote drops while transmitting data
|
|
struct Sender {
|
|
inner: Option<mpsc::UnboundedSender<Packet>>,
|
|
}
|
|
|
|
impl Sender {
|
|
fn new(inner: mpsc::UnboundedSender<Packet>) -> Self {
|
|
Sender { inner: Some(inner) }
|
|
}
|
|
|
|
fn send(&self, packet: Packet) {
|
|
let _ = self.inner.as_ref().unwrap().send(packet);
|
|
}
|
|
|
|
fn end(&mut self) {
|
|
self.inner = None;
|
|
}
|
|
}
|
|
|
|
impl Drop for Sender {
|
|
fn drop(&mut self) {
|
|
if let Some(inner) = self.inner.take() {
|
|
let _ = inner.send(Err(std::io::Error::new(
|
|
std::io::ErrorKind::BrokenPipe,
|
|
"Netapp connection dropped before end of stream",
|
|
)));
|
|
}
|
|
}
|
|
}
|
|
|
|
/// The RecvLoop trait, which is implemented both by the client and the server
|
|
/// connection objects (ServerConn and ClientConn) adds a method `.recv_loop()`
|
|
/// and a prototype of a handler for received messages `.recv_handler()` that
|
|
/// must be filled by implementors. `.recv_loop()` receives messages in a loop
|
|
/// according to the protocol defined above: chunks of message in progress of being
|
|
/// received are stored in a buffer, and when the last chunk of a message is received,
|
|
/// the full message is passed to the receive handler.
|
|
#[async_trait]
|
|
pub(crate) trait RecvLoop: Sync + 'static {
|
|
fn recv_handler(self: &Arc<Self>, id: RequestID, stream: ByteStream);
|
|
fn cancel_handler(self: &Arc<Self>, _id: RequestID) {}
|
|
|
|
async fn recv_loop<R>(self: Arc<Self>, mut read: R, debug_name: String) -> Result<(), Error>
|
|
where
|
|
R: AsyncReadExt + Unpin + Send + Sync,
|
|
{
|
|
let mut streams: HashMap<RequestID, Sender> = HashMap::new();
|
|
loop {
|
|
trace!(
|
|
"recv_loop({}): in_progress = {:?}",
|
|
debug_name,
|
|
streams.iter().map(|(id, _)| id).collect::<Vec<_>>()
|
|
);
|
|
|
|
let mut header_id = [0u8; RequestID::BITS as usize / 8];
|
|
match read.read_exact(&mut header_id[..]).await {
|
|
Ok(_) => (),
|
|
Err(e) if e.kind() == std::io::ErrorKind::UnexpectedEof => break,
|
|
Err(e) => return Err(e.into()),
|
|
};
|
|
let id = RequestID::from_be_bytes(header_id);
|
|
|
|
let mut header_size = [0u8; ChunkLength::BITS as usize / 8];
|
|
read.read_exact(&mut header_size[..]).await?;
|
|
let size = ChunkLength::from_be_bytes(header_size);
|
|
|
|
if size == CANCEL_REQUEST {
|
|
if let Some(mut stream) = streams.remove(&id) {
|
|
let _ = stream.send(Err(std::io::Error::new(
|
|
std::io::ErrorKind::Other,
|
|
"netapp: cancel requested",
|
|
)));
|
|
stream.end();
|
|
}
|
|
self.cancel_handler(id);
|
|
continue;
|
|
}
|
|
|
|
let has_cont = (size & CHUNK_FLAG_HAS_CONTINUATION) != 0;
|
|
let is_error = (size & CHUNK_FLAG_ERROR) != 0;
|
|
let size = (size & CHUNK_LENGTH_MASK) as usize;
|
|
let mut next_slice = vec![0; size as usize];
|
|
read.read_exact(&mut next_slice[..]).await?;
|
|
|
|
let packet = if is_error {
|
|
let kind = u8_to_io_errorkind(next_slice[0]);
|
|
let msg =
|
|
std::str::from_utf8(&next_slice[1..]).unwrap_or("<invalid utf8 error message>");
|
|
debug!(
|
|
"recv_loop({}): got id {}, error {:?}: {}",
|
|
debug_name, id, kind, msg
|
|
);
|
|
Some(Err(std::io::Error::new(kind, msg.to_string())))
|
|
} else {
|
|
trace!(
|
|
"recv_loop({}): got id {}, size {}, has_cont {}",
|
|
debug_name,
|
|
id,
|
|
size,
|
|
has_cont
|
|
);
|
|
if !next_slice.is_empty() {
|
|
Some(Ok(Bytes::from(next_slice)))
|
|
} else {
|
|
None
|
|
}
|
|
};
|
|
|
|
let mut sender = if let Some(send) = streams.remove(&(id)) {
|
|
send
|
|
} else {
|
|
let (send, recv) = mpsc::unbounded_channel();
|
|
trace!("recv_loop({}): id {} is new channel", debug_name, id);
|
|
self.recv_handler(
|
|
id,
|
|
Box::pin(tokio_stream::wrappers::UnboundedReceiverStream::new(recv)),
|
|
);
|
|
Sender::new(send)
|
|
};
|
|
|
|
if let Some(packet) = packet {
|
|
// If we cannot put packet in channel, it means that the
|
|
// receiving end of the channel is disconnected.
|
|
// We still need to reach eos before dropping this sender
|
|
let _ = sender.send(packet);
|
|
}
|
|
|
|
if has_cont {
|
|
assert!(!is_error);
|
|
streams.insert(id, sender);
|
|
} else {
|
|
trace!("recv_loop({}): close channel id {}", debug_name, id);
|
|
sender.end();
|
|
}
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|