generated from PlexSheep/rs-base
tokio is hard work
cargo devel CI / cargo CI (push) Successful in 2m18s
Details
cargo devel CI / cargo CI (push) Successful in 2m18s
Details
This commit is contained in:
parent
a11618add4
commit
13745e96b8
|
@ -18,8 +18,8 @@ clap = "4.4.18"
|
||||||
clap-num = "1.0.2"
|
clap-num = "1.0.2"
|
||||||
clap-verbosity-flag = "2.1.2"
|
clap-verbosity-flag = "2.1.2"
|
||||||
libpt = { version = "0.3.10", features = ["net"] }
|
libpt = { version = "0.3.10", features = ["net"] }
|
||||||
mio = { version = "0.8.10", features = ["net", "os-poll"] }
|
|
||||||
threadpool = { version = "1.8.1", optional = true }
|
threadpool = { version = "1.8.1", optional = true }
|
||||||
|
tokio = { version = "1.35.1", features = ["net", "rt", "macros"] }
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
default = ["server"]
|
default = ["server"]
|
||||||
|
|
|
@ -1,6 +1,9 @@
|
||||||
use crate::common::args::Cli;
|
use crate::common::args::Cli;
|
||||||
use clap::ValueEnum;
|
use clap::ValueEnum;
|
||||||
use std::fmt::Display;
|
use std::{fmt::Display, time::Duration};
|
||||||
|
|
||||||
|
const DEFAULT_TIMEOUT_LEN: u64 = 2000; // ms
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub enum Mode {
|
pub enum Mode {
|
||||||
Tcp,
|
Tcp,
|
||||||
|
@ -41,6 +44,7 @@ pub struct Config {
|
||||||
pub addr: std::net::SocketAddr,
|
pub addr: std::net::SocketAddr,
|
||||||
pub mode: Mode,
|
pub mode: Mode,
|
||||||
pub threads: usize,
|
pub threads: usize,
|
||||||
|
pub timeout: Duration,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Config {
|
impl Config {
|
||||||
|
@ -49,6 +53,7 @@ impl Config {
|
||||||
addr: cli.addr.clone(),
|
addr: cli.addr.clone(),
|
||||||
mode: cli.mode.clone(),
|
mode: cli.mode.clone(),
|
||||||
threads: cli.threads,
|
threads: cli.threads,
|
||||||
|
timeout: Duration::from_millis(DEFAULT_TIMEOUT_LEN)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
|
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use libpt::log::*;
|
use libpt::log::*;
|
||||||
|
use tokio;
|
||||||
|
|
||||||
mod client;
|
mod client;
|
||||||
mod common;
|
mod common;
|
||||||
|
@ -14,7 +15,8 @@ use common::{args::Cli, conf::*};
|
||||||
|
|
||||||
use crate::server::Server;
|
use crate::server::Server;
|
||||||
|
|
||||||
fn main() -> Result<()> {
|
#[tokio::main]
|
||||||
|
async fn main() -> Result<()> {
|
||||||
let cli = Cli::cli_parse();
|
let cli = Cli::cli_parse();
|
||||||
debug!("dumping cli args:\n{:#?}", cli);
|
debug!("dumping cli args:\n{:#?}", cli);
|
||||||
|
|
||||||
|
@ -23,8 +25,7 @@ fn main() -> Result<()> {
|
||||||
#[cfg(feature = "server")]
|
#[cfg(feature = "server")]
|
||||||
if cli.server {
|
if cli.server {
|
||||||
info!("starting server");
|
info!("starting server");
|
||||||
Server::build(cfg)?.run()?;
|
return Server::build(cfg).await?.run().await
|
||||||
return Ok(());
|
|
||||||
}
|
}
|
||||||
// implicit else, so we can work without the server feature
|
// implicit else, so we can work without the server feature
|
||||||
info!("starting client");
|
info!("starting client");
|
||||||
|
|
|
@ -1,69 +1,106 @@
|
||||||
#![cfg(feature = "server")]
|
#![cfg(feature = "server")]
|
||||||
use anyhow::Result;
|
use std::{net::SocketAddr, time::Duration};
|
||||||
use libpt::log::{error, info};
|
|
||||||
use mio::{self, event::Event, net::TcpListener, Events, Interest, Poll, Token};
|
use anyhow::{Result, anyhow};
|
||||||
use std::time::Duration;
|
use libpt::{
|
||||||
|
bintols::display::humanbytes,
|
||||||
|
log::{error, info, trace, warn, debug},
|
||||||
|
};
|
||||||
use threadpool::ThreadPool;
|
use threadpool::ThreadPool;
|
||||||
|
use tokio::{
|
||||||
|
io::{AsyncBufRead, AsyncBufReadExt, AsyncReadExt, AsyncWriteExt, BufReader, Interest},
|
||||||
|
net::{self, TcpListener, TcpSocket, TcpStream},
|
||||||
|
select,
|
||||||
|
time::timeout,
|
||||||
|
};
|
||||||
|
|
||||||
use crate::common::conf::Config;
|
use crate::common::conf::Config;
|
||||||
|
|
||||||
pub mod listener;
|
pub mod listener;
|
||||||
|
|
||||||
const EVENT_CAPACITY: usize = 128;
|
|
||||||
const SERVER: Token = Token(0);
|
|
||||||
|
|
||||||
pub struct Server {
|
pub struct Server {
|
||||||
cfg: Config,
|
cfg: Config,
|
||||||
work: Poll,
|
|
||||||
events: Events,
|
|
||||||
pool: ThreadPool,
|
pool: ThreadPool,
|
||||||
pub timeout: Option<Duration>,
|
pub timeout: Option<Duration>,
|
||||||
server: TcpListener,
|
server: TcpListener,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Server {
|
impl Server {
|
||||||
pub fn build(cfg: Config) -> Result<Self> {
|
pub async fn build(cfg: Config) -> Result<Self> {
|
||||||
let mut server = TcpListener::bind(cfg.addr)?;
|
let server = TcpListener::bind(cfg.addr).await?;
|
||||||
let poll: Poll = Poll::new()?;
|
|
||||||
poll.registry()
|
|
||||||
.register(&mut server, SERVER, Interest::READABLE | Interest::WRITABLE);
|
|
||||||
let events: Events = Events::with_capacity(EVENT_CAPACITY);
|
|
||||||
let pool = ThreadPool::new(cfg.threads);
|
let pool = ThreadPool::new(cfg.threads);
|
||||||
let timeout = Some(Duration::from_secs(5));
|
let timeout = Some(Duration::from_secs(5));
|
||||||
Ok(Server {
|
Ok(Server {
|
||||||
cfg,
|
cfg,
|
||||||
work: poll,
|
|
||||||
events,
|
|
||||||
pool,
|
pool,
|
||||||
timeout,
|
timeout,
|
||||||
server,
|
server,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
pub fn run(&mut self) -> Result<()> {
|
pub async fn run(self) -> Result<()> {
|
||||||
loop {
|
loop {
|
||||||
self.work.poll(&mut self.events, self.timeout)?;
|
let stream = self.server.accept().await?;
|
||||||
for event in self.events.iter() {
|
self.handle_stream(stream).await?;
|
||||||
let result = self.handle_event(event);
|
|
||||||
if let Err(err) = result {
|
|
||||||
error!("Error while handling server event {:?}: {:?}", event, err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handle_event(&self, event: &Event) -> Result<()> {
|
async fn handle_stream(&self, mut stream: (TcpStream, SocketAddr)) -> Result<()> {
|
||||||
dbg!(event);
|
info!("start handling stream {:?}", stream.1);
|
||||||
let connection = match self.server.accept() {
|
let mut buf = Vec::new();
|
||||||
Ok(c) => c,
|
let mut reader = BufReader::new(stream.0);
|
||||||
Err(err) => {
|
let mut len;
|
||||||
error!("error while handling connection: {:?}", err);
|
loop {
|
||||||
return Err(err.into());
|
trace!("reading anew");
|
||||||
|
len = self.read(&mut reader, &mut buf).await?;
|
||||||
|
trace!("did read");
|
||||||
|
if len == 0 {
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
info!(
|
||||||
|
"< {}\t({})\n{}",
|
||||||
|
stream.1,
|
||||||
|
humanbytes(len),
|
||||||
|
self.decode(&buf)?
|
||||||
|
);
|
||||||
|
buf.clear();
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
self.pool.execute(move|| {
|
info!("stop handling stream {:?}", stream.1);
|
||||||
dbg!(connection);
|
|
||||||
info!("received a connection!");
|
|
||||||
});
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn decode(&self, buf: &Vec<u8>) -> Result<String> {
|
||||||
|
Ok(String::from_utf8(buf.clone())?)
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn read(&self, reader: &mut BufReader<TcpStream>, buf: &mut Vec<u8>) -> Result<usize> {
|
||||||
|
let mut len: usize;
|
||||||
|
let mut failsafe: u8 = 0;
|
||||||
|
while failsafe < u8::MAX {
|
||||||
|
trace!("loop");
|
||||||
|
len = match timeout(self.cfg.timeout, reader.read(buf)).await {
|
||||||
|
Ok(inner) => {
|
||||||
|
trace!("Read for len: {inner:?}"); 0
|
||||||
|
},
|
||||||
|
Err(err) => {
|
||||||
|
error!("timeout while reading: {err}");
|
||||||
|
return Err(err.into());
|
||||||
|
}
|
||||||
|
};
|
||||||
|
if self.should_end(&buf) {
|
||||||
|
return Ok(len);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// ?????????
|
||||||
|
failsafe += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(anyhow!("read too often, so the failsafe activated"))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn should_end(&self, buf: &Vec<u8>) -> bool {
|
||||||
|
debug!("eval should end: {:?}", buf);
|
||||||
|
buf.contains(&0x00)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue