mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-04 13:27:39 +03:00
193 lines
5.2 KiB
Rust
193 lines
5.2 KiB
Rust
use std::{
|
|
any::Any, any::TypeId, fmt, future::Future, io as sio, io::Error as IoError,
|
|
task::Context, task::Poll,
|
|
};
|
|
|
|
pub mod testing;
|
|
pub mod types;
|
|
|
|
mod dispatcher;
|
|
mod filter;
|
|
mod io;
|
|
mod ioref;
|
|
mod seal;
|
|
mod tasks;
|
|
mod time;
|
|
mod utils;
|
|
|
|
#[cfg(any(feature = "tokio-traits", feature = "tokio"))]
|
|
mod tokio_impl;
|
|
#[cfg(any(feature = "tokio"))]
|
|
mod tokio_rt;
|
|
|
|
use ntex_bytes::BytesMut;
|
|
use ntex_codec::{Decoder, Encoder};
|
|
use ntex_util::time::Millis;
|
|
|
|
pub use self::dispatcher::Dispatcher;
|
|
pub use self::filter::Base;
|
|
pub use self::io::{Io, IoRef, OnDisconnect};
|
|
pub use self::seal::{IoBoxed, Sealed};
|
|
pub use self::tasks::{ReadContext, WriteContext};
|
|
pub use self::time::Timer;
|
|
pub use self::utils::{add_filter, boxed, seal, Boxed, BoxedFactory};
|
|
|
|
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
|
|
pub enum ReadStatus {
|
|
Ready,
|
|
Terminate,
|
|
}
|
|
|
|
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
|
|
pub enum WriteStatus {
|
|
Ready,
|
|
Timeout(Millis),
|
|
Shutdown(Millis),
|
|
Terminate,
|
|
}
|
|
|
|
pub trait Filter: 'static {
|
|
fn query(&self, id: TypeId) -> Option<Box<dyn Any>>;
|
|
|
|
/// Filter needs incoming data from io stream
|
|
fn want_read(&self);
|
|
|
|
/// Filter wants gracefully shutdown io stream
|
|
fn want_shutdown(&self);
|
|
|
|
fn poll_shutdown(&self) -> Poll<sio::Result<()>>;
|
|
|
|
fn poll_read_ready(&self, cx: &mut Context<'_>) -> Poll<ReadStatus>;
|
|
|
|
fn poll_write_ready(&self, cx: &mut Context<'_>) -> Poll<WriteStatus>;
|
|
|
|
fn get_read_buf(&self) -> Option<BytesMut>;
|
|
|
|
fn get_write_buf(&self) -> Option<BytesMut>;
|
|
|
|
fn release_read_buf(
|
|
&self,
|
|
src: BytesMut,
|
|
dst: &mut Option<BytesMut>,
|
|
nbytes: usize,
|
|
) -> sio::Result<usize>;
|
|
|
|
fn release_write_buf(&self, buf: BytesMut) -> sio::Result<()>;
|
|
|
|
fn closed(&self, err: Option<sio::Error>);
|
|
}
|
|
|
|
pub trait FilterFactory<F: Filter>: Sized {
|
|
type Filter: Filter;
|
|
|
|
type Error: fmt::Debug;
|
|
type Future: Future<Output = Result<Io<Self::Filter>, Self::Error>>;
|
|
|
|
fn create(self, st: Io<F>) -> Self::Future;
|
|
}
|
|
|
|
pub trait IoStream {
|
|
fn start(self, _: ReadContext, _: WriteContext) -> Option<Box<dyn Handle>>;
|
|
}
|
|
|
|
pub trait Handle {
|
|
fn query(&self, id: TypeId) -> Option<Box<dyn Any>>;
|
|
}
|
|
|
|
/// Recv error
|
|
#[derive(Debug)]
|
|
pub enum RecvError<U: Decoder> {
|
|
/// Keep-alive timeout occured
|
|
KeepAlive,
|
|
/// Write backpressure is enabled
|
|
WriteBackpressure,
|
|
/// Dispatcher marked stopped
|
|
StopDispatcher,
|
|
/// Unrecoverable frame decoding errors
|
|
Decoder(U::Error),
|
|
/// Peer is disconnected
|
|
PeerGone(Option<sio::Error>),
|
|
}
|
|
|
|
/// Dispatcher item
|
|
pub enum DispatchItem<U: Encoder + Decoder> {
|
|
Item(<U as Decoder>::Item),
|
|
/// Write back-pressure enabled
|
|
WBackPressureEnabled,
|
|
/// Write back-pressure disabled
|
|
WBackPressureDisabled,
|
|
/// Keep alive timeout
|
|
KeepAliveTimeout,
|
|
/// Decoder parse error
|
|
DecoderError(<U as Decoder>::Error),
|
|
/// Encoder parse error
|
|
EncoderError(<U as Encoder>::Error),
|
|
/// Socket is disconnected
|
|
Disconnect(Option<IoError>),
|
|
}
|
|
|
|
impl<U> fmt::Debug for DispatchItem<U>
|
|
where
|
|
U: Encoder + Decoder,
|
|
<U as Decoder>::Item: fmt::Debug,
|
|
{
|
|
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
match *self {
|
|
DispatchItem::Item(ref item) => {
|
|
write!(fmt, "DispatchItem::Item({:?})", item)
|
|
}
|
|
DispatchItem::WBackPressureEnabled => {
|
|
write!(fmt, "DispatchItem::WBackPressureEnabled")
|
|
}
|
|
DispatchItem::WBackPressureDisabled => {
|
|
write!(fmt, "DispatchItem::WBackPressureDisabled")
|
|
}
|
|
DispatchItem::KeepAliveTimeout => {
|
|
write!(fmt, "DispatchItem::KeepAliveTimeout")
|
|
}
|
|
DispatchItem::EncoderError(ref e) => {
|
|
write!(fmt, "DispatchItem::EncoderError({:?})", e)
|
|
}
|
|
DispatchItem::DecoderError(ref e) => {
|
|
write!(fmt, "DispatchItem::DecoderError({:?})", e)
|
|
}
|
|
DispatchItem::Disconnect(ref e) => {
|
|
write!(fmt, "DispatchItem::Disconnect({:?})", e)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub mod rt {
|
|
//! async runtime helpers
|
|
|
|
#[cfg(feature = "tokio")]
|
|
pub use crate::tokio_rt::*;
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use ntex_codec::BytesCodec;
|
|
use std::io;
|
|
|
|
#[test]
|
|
fn test_fmt() {
|
|
type T = DispatchItem<BytesCodec>;
|
|
|
|
let err = T::EncoderError(io::Error::new(io::ErrorKind::Other, "err"));
|
|
assert!(format!("{:?}", err).contains("DispatchItem::Encoder"));
|
|
let err = T::DecoderError(io::Error::new(io::ErrorKind::Other, "err"));
|
|
assert!(format!("{:?}", err).contains("DispatchItem::Decoder"));
|
|
let err = T::Disconnect(Some(io::Error::new(io::ErrorKind::Other, "err")));
|
|
assert!(format!("{:?}", err).contains("DispatchItem::Disconnect"));
|
|
|
|
assert!(format!("{:?}", T::WBackPressureEnabled)
|
|
.contains("DispatchItem::WBackPressureEnabled"));
|
|
assert!(format!("{:?}", T::WBackPressureDisabled)
|
|
.contains("DispatchItem::WBackPressureDisabled"));
|
|
assert!(
|
|
format!("{:?}", T::KeepAliveTimeout).contains("DispatchItem::KeepAliveTimeout")
|
|
);
|
|
}
|
|
}
|