use crate::network::adapter::{
Resource, Remote, Local, Adapter, SendStatus, AcceptedType, ReadStatus, ConnectionInfo,
ListeningInfo,
};
use crate::network::{RemoteAddr};
use crate::util::encoding::{self, Decoder, MAX_ENCODED_SIZE};
use mio::net::{TcpListener, TcpStream};
use mio::event::{Source};
use std::net::{SocketAddr, TcpStream as StdTcpStream};
use std::io::{self, ErrorKind, Read, Write};
use std::ops::{Deref};
use std::cell::{RefCell};
use std::mem::{MaybeUninit};
const INPUT_BUFFER_SIZE: usize = u16::MAX as usize;
pub(crate) struct FramedTcpAdapter;
impl Adapter for FramedTcpAdapter {
type Remote = RemoteResource;
type Local = LocalResource;
}
pub(crate) struct RemoteResource {
stream: TcpStream,
decoder: RefCell<Decoder>,
}
unsafe impl Sync for RemoteResource {}
impl From<TcpStream> for RemoteResource {
fn from(stream: TcpStream) -> Self {
Self { stream, decoder: RefCell::new(Decoder::default()) }
}
}
impl Resource for RemoteResource {
fn source(&mut self) -> &mut dyn Source {
&mut self.stream
}
}
impl Remote for RemoteResource {
fn connect(remote_addr: RemoteAddr) -> io::Result<ConnectionInfo<Self>> {
let peer_addr = *remote_addr.socket_addr();
let stream = StdTcpStream::connect(peer_addr)?;
let local_addr = stream.local_addr()?;
stream.set_nonblocking(true)?;
Ok(ConnectionInfo { remote: TcpStream::from_std(stream).into(), local_addr, peer_addr })
}
fn receive(&self, mut process_data: impl FnMut(&[u8])) -> ReadStatus {
let buffer: MaybeUninit<[u8; INPUT_BUFFER_SIZE]> = MaybeUninit::uninit();
let mut input_buffer = unsafe { buffer.assume_init() };
loop {
let stream = &self.stream;
match stream.deref().read(&mut input_buffer) {
Ok(0) => break ReadStatus::Disconnected,
Ok(size) => {
let data = &input_buffer[..size];
let addr = self.stream.peer_addr().unwrap();
log::trace!("Decoding data from {}, {} bytes", addr, data.len());
self.decoder.borrow_mut().decode(data, |decoded_data| {
process_data(decoded_data);
});
}
Err(ref err) if err.kind() == ErrorKind::Interrupted => continue,
Err(ref err) if err.kind() == ErrorKind::WouldBlock => {
break ReadStatus::WaitNextEvent
}
Err(ref err) if err.kind() == ErrorKind::ConnectionReset => {
break ReadStatus::Disconnected
}
Err(err) => {
log::error!("TCP receive error: {}", err);
break ReadStatus::Disconnected }
}
}
}
fn send(&self, data: &[u8]) -> SendStatus {
let mut buf = [0; MAX_ENCODED_SIZE]; let encoded_size = encoding::encode_size(data, &mut buf);
let mut total_bytes_sent = 0;
let total_bytes = encoded_size.len() + data.len();
loop {
let data_to_send = match total_bytes_sent < encoded_size.len() {
true => &encoded_size[total_bytes_sent..],
false => &data[total_bytes_sent - encoded_size.len()..],
};
let stream = &self.stream;
match stream.deref().write(data_to_send) {
Ok(bytes_sent) => {
total_bytes_sent += bytes_sent;
if total_bytes_sent == total_bytes {
break SendStatus::Sent
}
}
Err(ref err) if err.kind() == io::ErrorKind::WouldBlock => continue,
Err(err) => {
log::error!("TCP receive error: {}", err);
break SendStatus::ResourceNotFound }
}
}
}
}
pub(crate) struct LocalResource {
listener: TcpListener,
}
impl Resource for LocalResource {
fn source(&mut self) -> &mut dyn Source {
&mut self.listener
}
}
impl Local for LocalResource {
type Remote = RemoteResource;
fn listen(addr: SocketAddr) -> io::Result<ListeningInfo<Self>> {
let listener = TcpListener::bind(addr)?;
let local_addr = listener.local_addr().unwrap();
Ok(ListeningInfo { local: { LocalResource { listener } }, local_addr })
}
fn accept(&self, mut accept_remote: impl FnMut(AcceptedType<'_, Self::Remote>)) {
loop {
match self.listener.accept() {
Ok((stream, addr)) => accept_remote(AcceptedType::Remote(addr, stream.into())),
Err(ref err) if err.kind() == ErrorKind::WouldBlock => break,
Err(ref err) if err.kind() == ErrorKind::Interrupted => continue,
Err(err) => break log::error!("TCP accept error: {}", err), }
}
}
}