diff --git a/Cargo.lock b/Cargo.lock index 49954d451..fe46e32d6 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -781,7 +781,6 @@ dependencies = [ "thiserror", "tokio", "tokio-boring", - "tokio-io-timeout", "tokio-rustls", "tokio-test", "tokio-tungstenite", diff --git a/clash_lib/Cargo.toml b/clash_lib/Cargo.toml index a60870cd1..2efb498dc 100644 --- a/clash_lib/Cargo.toml +++ b/clash_lib/Cargo.toml @@ -12,7 +12,6 @@ bench = ["criterion"] tokio = { version = "1", features = ["full"] } tokio-util = { version = "0.7", features = ["net", "codec", "io"] } tokio-rustls = "0.24" -tokio-io-timeout = "1.2.0" thiserror = "1.0" async-trait = "0.1" anyhow = "1.0" diff --git a/clash_lib/src/app/dispatcher/dispatcher.rs b/clash_lib/src/app/dispatcher/dispatcher.rs index 8c76a6cde..a2fc86e7f 100644 --- a/clash_lib/src/app/dispatcher/dispatcher.rs +++ b/clash_lib/src/app/dispatcher/dispatcher.rs @@ -2,6 +2,7 @@ use crate::app::dispatcher::tracked::TrackedDatagram; use crate::app::dispatcher::tracked::TrackedStream; use crate::app::outbound::manager::ThreadSafeOutboundManager; use crate::app::router::ThreadSafeRouter; +use crate::common::io::copy_buf_bidirectional_with_timeout; use crate::config::def::RunMode; use crate::config::internal::proxy::PROXY_DIRECT; use crate::config::internal::proxy::PROXY_GLOBAL; @@ -17,10 +18,9 @@ use std::net::SocketAddr; use std::sync::Arc; use std::time::Duration; use std::time::Instant; -use tokio::io::{copy_bidirectional, AsyncRead, AsyncWrite, AsyncWriteExt}; +use tokio::io::{AsyncRead, AsyncWrite, AsyncWriteExt}; use tokio::sync::RwLock; use tokio::task::JoinHandle; -use tokio_io_timeout::TimeoutStream; use tracing::info_span; use tracing::instrument; use tracing::trace; @@ -132,20 +132,21 @@ impl Dispatcher { { Ok(rhs) => { debug!("remote connection established {}", sess); - let rhs = TrackedStream::new(rhs, self.manager.clone(), sess.clone(), rule).await; - let mut rhs = TimeoutStream::new(rhs); - // let's close it if no data is transferred in 60 seconds - // plz do your own keepalive if you need it - rhs.set_read_timeout(Some(Duration::from_secs(60))); - rhs.set_write_timeout(Some(Duration::from_secs(60))); - let mut rhs = Box::pin(rhs); - match copy_bidirectional(&mut lhs, &mut rhs) - .instrument(info_span!( - "copy_bidirectional", - outbound_name = outbound_name, - session = %sess, - )) - .await + let mut rhs = + TrackedStream::new(rhs, self.manager.clone(), sess.clone(), rule).await; + match copy_buf_bidirectional_with_timeout( + &mut lhs, + &mut rhs, + 4096, + Duration::from_secs(10), + Duration::from_secs(10), + ) + .instrument(info_span!( + "copy_bidirectional", + outbound_name = outbound_name, + session = %sess, + )) + .await { Ok((up, down)) => { debug!( diff --git a/clash_lib/src/app/dispatcher/tracked.rs b/clash_lib/src/app/dispatcher/tracked.rs index 070a70881..fcad977a4 100644 --- a/clash_lib/src/app/dispatcher/tracked.rs +++ b/clash_lib/src/app/dispatcher/tracked.rs @@ -195,12 +195,6 @@ impl AsyncRead for TrackedStream { .download_total .fetch_add(download as u64, std::sync::atomic::Ordering::Release); - // notify the copy_bidirectional to close - if download == 0 { - debug!("connection closed by peer: {}", self.id()); - return Poll::Ready(Err(std::io::ErrorKind::BrokenPipe.into())); - } - v } } @@ -231,12 +225,6 @@ impl AsyncWrite for TrackedStream { .upload_total .fetch_add(upload as u64, std::sync::atomic::Ordering::Release); - // notify the copy_bidirectional to close - if upload == 0 { - debug!("connection closed by peer: {}", self.id()); - return Poll::Ready(Err(std::io::ErrorKind::BrokenPipe.into())); - } - v } diff --git a/clash_lib/src/common/io.rs b/clash_lib/src/common/io.rs new file mode 100644 index 000000000..b9e5fa59b --- /dev/null +++ b/clash_lib/src/common/io.rs @@ -0,0 +1,291 @@ +/// copy of https://github.com/eycorsican/leaf/blob/a77a1e497ae034f3a2a89c8628d5e7ebb2af47f0/leaf/src/common/io.rs +use std::future::Future; +use std::io; +use std::pin::Pin; +use std::task::{Context, Poll}; +use std::time::Duration; + +use futures::ready; +use tokio::io::{AsyncRead, AsyncWrite, ReadBuf}; + +#[derive(Debug)] +pub struct CopyBuffer { + read_done: bool, + need_flush: bool, + pos: usize, + cap: usize, + amt: u64, + buf: Box<[u8]>, +} + +impl CopyBuffer { + #[allow(unused)] + pub fn new() -> Self { + Self { + read_done: false, + need_flush: false, + pos: 0, + cap: 0, + amt: 0, + buf: vec![0; 2 * 1024].into_boxed_slice(), + } + } + + pub fn new_with_capacity(size: usize) -> Result { + let mut buf = Vec::new(); + buf.try_reserve(size).map_err(|e| { + std::io::Error::new( + std::io::ErrorKind::Other, + format!("new buffer failed: {}", e), + ) + })?; + buf.resize(size, 0); + Ok(Self { + read_done: false, + need_flush: false, + pos: 0, + cap: 0, + amt: 0, + buf: buf.into_boxed_slice(), + }) + } + + pub fn amount_transfered(&self) -> u64 { + self.amt + } + + pub fn poll_copy( + &mut self, + cx: &mut Context<'_>, + mut reader: Pin<&mut R>, + mut writer: Pin<&mut W>, + ) -> Poll> + where + R: AsyncRead + ?Sized, + W: AsyncWrite + ?Sized, + { + loop { + // If our buffer is empty, then we need to read some data to + // continue. + if self.pos == self.cap && !self.read_done { + let me = &mut *self; + let mut buf = ReadBuf::new(&mut me.buf); + + match reader.as_mut().poll_read(cx, &mut buf) { + Poll::Ready(Ok(_)) => (), + Poll::Ready(Err(err)) => return Poll::Ready(Err(err)), + Poll::Pending => { + // Try flushing when the reader has no progress to avoid deadlock + // when the reader depends on buffered writer. + if self.need_flush { + ready!(writer.as_mut().poll_flush(cx))?; + self.need_flush = false; + } + + return Poll::Pending; + } + } + + let n = buf.filled().len(); + if n == 0 { + self.read_done = true; + } else { + self.pos = 0; + self.cap = n; + } + } + + // If our buffer has some data, let's write it out! + while self.pos < self.cap { + let me = &mut *self; + let i = ready!(writer.as_mut().poll_write(cx, &me.buf[me.pos..me.cap]))?; + if i == 0 { + return Poll::Ready(Err(io::Error::new( + io::ErrorKind::WriteZero, + "write zero byte into writer", + ))); + } else { + self.pos += i; + self.amt += i as u64; + self.need_flush = true; + } + } + + // If pos larger than cap, this loop will never stop. + // In particular, user's wrong poll_write implementation returning + // incorrect written length may lead to thread blocking. + debug_assert!( + self.pos <= self.cap, + "writer returned length larger than input slice" + ); + + // If we've written all the data and we've seen EOF, flush out the + // data and finish the transfer. + if self.pos == self.cap && self.read_done { + ready!(writer.as_mut().poll_flush(cx))?; + return Poll::Ready(Ok(self.amt)); + } + } + } +} + +enum TransferState { + Running(CopyBuffer), + ShuttingDown(u64), + Done, +} + +struct CopyBidirectional<'a, A: ?Sized, B: ?Sized> { + a: &'a mut A, + b: &'a mut B, + a_to_b: TransferState, + b_to_a: TransferState, + a_to_b_count: u64, + b_to_a_count: u64, + a_to_b_delay: Option>>, + b_to_a_delay: Option>>, + a_to_b_timeout_duration: Duration, + b_to_a_timeout_duration: Duration, +} + +impl<'a, A, B> Future for CopyBidirectional<'a, A, B> +where + A: AsyncRead + AsyncWrite + Unpin + ?Sized, + B: AsyncRead + AsyncWrite + Unpin + ?Sized, +{ + type Output = io::Result<(u64, u64)>; + + fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { + // Unpack self into mut refs to each field to avoid borrow check issues. + let CopyBidirectional { + a, + b, + a_to_b, + b_to_a, + a_to_b_count, + b_to_a_count, + a_to_b_delay, + b_to_a_delay, + a_to_b_timeout_duration, + b_to_a_timeout_duration, + } = &mut *self; + + let mut a = Pin::new(a); + let mut b = Pin::new(b); + + loop { + match a_to_b { + TransferState::Running(buf) => { + let res = buf.poll_copy(cx, a.as_mut(), b.as_mut()); + match res { + Poll::Ready(Ok(count)) => { + *a_to_b = TransferState::ShuttingDown(count); + continue; + } + Poll::Ready(Err(err)) => return Poll::Ready(Err(err)), + Poll::Pending => { + if let Some(delay) = a_to_b_delay { + match delay.as_mut().poll(cx) { + Poll::Ready(()) => { + *a_to_b = + TransferState::ShuttingDown(buf.amount_transfered()); + continue; + } + Poll::Pending => (), + } + } + } + } + } + TransferState::ShuttingDown(count) => { + let res = b.as_mut().poll_shutdown(cx); + match res { + Poll::Ready(Ok(())) => { + *a_to_b_count += *count; + *a_to_b = TransferState::Done; + b_to_a_delay + .replace(Box::pin(tokio::time::sleep(*b_to_a_timeout_duration))); + continue; + } + Poll::Ready(Err(err)) => return Poll::Ready(Err(err)), + Poll::Pending => (), + } + } + TransferState::Done => (), + } + + match b_to_a { + TransferState::Running(buf) => { + let res = buf.poll_copy(cx, b.as_mut(), a.as_mut()); + match res { + Poll::Ready(Ok(count)) => { + *b_to_a = TransferState::ShuttingDown(count); + continue; + } + Poll::Ready(Err(err)) => return Poll::Ready(Err(err)), + Poll::Pending => { + if let Some(delay) = b_to_a_delay { + match delay.as_mut().poll(cx) { + Poll::Ready(()) => { + *b_to_a = + TransferState::ShuttingDown(buf.amount_transfered()); + continue; + } + Poll::Pending => (), + } + } + } + } + } + TransferState::ShuttingDown(count) => { + let res = a.as_mut().poll_shutdown(cx); + match res { + Poll::Ready(Ok(())) => { + *b_to_a_count += *count; + *b_to_a = TransferState::Done; + a_to_b_delay + .replace(Box::pin(tokio::time::sleep(*a_to_b_timeout_duration))); + continue; + } + Poll::Ready(Err(err)) => return Poll::Ready(Err(err)), + Poll::Pending => (), + } + } + TransferState::Done => (), + } + + match (&a_to_b, &b_to_a) { + (TransferState::Done, TransferState::Done) => break, + _ => return Poll::Pending, + } + } + + Poll::Ready(Ok((*a_to_b_count, *b_to_a_count))) + } +} + +pub async fn copy_buf_bidirectional_with_timeout( + a: &mut A, + b: &mut B, + size: usize, + a_to_b_timeout_duration: Duration, + b_to_a_timeout_duration: Duration, +) -> Result<(u64, u64), std::io::Error> +where + A: AsyncRead + AsyncWrite + Unpin + ?Sized, + B: AsyncRead + AsyncWrite + Unpin + ?Sized, +{ + CopyBidirectional { + a, + b, + a_to_b: TransferState::Running(CopyBuffer::new_with_capacity(size)?), + b_to_a: TransferState::Running(CopyBuffer::new_with_capacity(size)?), + a_to_b_count: 0, + b_to_a_count: 0, + a_to_b_delay: None, + b_to_a_delay: None, + a_to_b_timeout_duration, + b_to_a_timeout_duration, + } + .await +} diff --git a/clash_lib/src/common/mod.rs b/clash_lib/src/common/mod.rs index 6ccdf8aab..3604b7d55 100644 --- a/clash_lib/src/common/mod.rs +++ b/clash_lib/src/common/mod.rs @@ -2,6 +2,7 @@ pub mod auth; pub mod crypto; pub mod errors; pub mod http; +pub mod io; pub mod mmdb; pub mod timed_future; pub mod tls;