mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-06 14:27:39 +03:00
101 lines
2.9 KiB
Rust
101 lines
2.9 KiB
Rust
use std::{any, future::poll_fn, task::Poll};
|
|
|
|
use ntex_io::{
|
|
types, Handle, IoContext, IoStream, ReadContext, ReadStatus, WriteContext, WriteStatus,
|
|
};
|
|
use ntex_rt::spawn;
|
|
use socket2::Socket;
|
|
|
|
use super::driver::{StreamCtl, StreamOps};
|
|
|
|
impl IoStream for super::TcpStream {
|
|
fn start(self, read: ReadContext, _: WriteContext) -> Option<Box<dyn Handle>> {
|
|
let io = self.0;
|
|
let context = read.context();
|
|
let ctl = StreamOps::current().register(io, context.clone());
|
|
let ctl2 = ctl.clone();
|
|
spawn(async move { run(ctl, context).await });
|
|
|
|
Some(Box::new(HandleWrapper(ctl2)))
|
|
}
|
|
}
|
|
|
|
impl IoStream for super::UnixStream {
|
|
fn start(self, read: ReadContext, _: WriteContext) -> Option<Box<dyn Handle>> {
|
|
let io = self.0;
|
|
let context = read.context();
|
|
let ctl = StreamOps::current().register(io, context.clone());
|
|
spawn(async move { run(ctl, context).await });
|
|
|
|
None
|
|
}
|
|
}
|
|
|
|
struct HandleWrapper(StreamCtl<Socket>);
|
|
|
|
impl Handle for HandleWrapper {
|
|
fn query(&self, id: any::TypeId) -> Option<Box<dyn any::Any>> {
|
|
if id == any::TypeId::of::<types::PeerAddr>() {
|
|
let addr = self.0.with_io(|io| io.and_then(|io| io.peer_addr().ok()));
|
|
if let Some(addr) = addr.and_then(|addr| addr.as_socket()) {
|
|
return Some(Box::new(types::PeerAddr(addr)));
|
|
}
|
|
}
|
|
None
|
|
}
|
|
}
|
|
|
|
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
|
enum Status {
|
|
Shutdown,
|
|
Terminate,
|
|
}
|
|
|
|
async fn run<T>(ctl: StreamCtl<T>, context: IoContext) {
|
|
// Handle io read readiness
|
|
let st = poll_fn(|cx| {
|
|
let mut modify = false;
|
|
let mut readable = false;
|
|
let mut writable = false;
|
|
let read = match context.poll_read_ready(cx) {
|
|
Poll::Ready(ReadStatus::Ready) => {
|
|
modify = true;
|
|
readable = true;
|
|
Poll::Pending
|
|
}
|
|
Poll::Ready(ReadStatus::Terminate) => Poll::Ready(()),
|
|
Poll::Pending => {
|
|
modify = true;
|
|
Poll::Pending
|
|
}
|
|
};
|
|
|
|
let write = match context.poll_write_ready(cx) {
|
|
Poll::Ready(WriteStatus::Ready) => {
|
|
modify = true;
|
|
writable = true;
|
|
Poll::Pending
|
|
}
|
|
Poll::Ready(WriteStatus::Shutdown) => Poll::Ready(Status::Shutdown),
|
|
Poll::Ready(WriteStatus::Terminate) => Poll::Ready(Status::Terminate),
|
|
Poll::Pending => Poll::Pending,
|
|
};
|
|
|
|
if modify {
|
|
ctl.modify(readable, writable);
|
|
}
|
|
|
|
if read.is_pending() && write.is_pending() {
|
|
Poll::Pending
|
|
} else if write.is_ready() {
|
|
write
|
|
} else {
|
|
Poll::Ready(Status::Terminate)
|
|
}
|
|
})
|
|
.await;
|
|
|
|
ctl.modify(false, true);
|
|
context.shutdown(st == Status::Shutdown).await;
|
|
context.stopped(ctl.close().await.err());
|
|
}
|