mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-04 13:27:39 +03:00
202 lines
5.8 KiB
Rust
202 lines
5.8 KiB
Rust
//! Utilities for abstructing io streams
|
|
#![allow(clippy::return_self_not_must_use)]
|
|
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 framed;
|
|
mod io;
|
|
mod ioref;
|
|
mod seal;
|
|
mod tasks;
|
|
mod timer;
|
|
mod utils;
|
|
|
|
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::framed::Framed;
|
|
pub use self::io::{Io, IoRef, OnDisconnect};
|
|
pub use self::seal::{IoBoxed, Sealed};
|
|
pub use self::tasks::{ReadContext, WriteContext};
|
|
pub use self::utils::{filter, seal};
|
|
|
|
/// Status for read task
|
|
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
|
|
pub enum ReadStatus {
|
|
Ready,
|
|
Terminate,
|
|
}
|
|
|
|
/// Status for write task
|
|
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
|
|
pub enum WriteStatus {
|
|
/// Write task is clear to proceed with write operation
|
|
Ready,
|
|
/// Initiate timeout for normal write operations, shutdown connection after timeout
|
|
Timeout(Millis),
|
|
/// Initiate graceful io shutdown operation with timeout
|
|
Shutdown(Millis),
|
|
/// Immediately terminate connection
|
|
Terminate,
|
|
}
|
|
|
|
pub trait Filter: 'static {
|
|
fn query(&self, id: TypeId) -> Option<Box<dyn Any>>;
|
|
|
|
fn get_read_buf(&self) -> Option<BytesMut>;
|
|
|
|
fn release_read_buf(&self, buf: BytesMut);
|
|
|
|
/// Process read buffer
|
|
///
|
|
/// Returns tuple (total bytes, new bytes)
|
|
fn process_read_buf(&self, io: &IoRef, n: usize) -> sio::Result<(usize, usize)>;
|
|
|
|
fn get_write_buf(&self) -> Option<BytesMut>;
|
|
|
|
fn release_write_buf(&self, buf: BytesMut) -> sio::Result<()>;
|
|
|
|
/// Check readiness for read operations
|
|
fn poll_read_ready(&self, cx: &mut Context<'_>) -> Poll<ReadStatus>;
|
|
|
|
/// Check readiness for write operations
|
|
fn poll_write_ready(&self, cx: &mut Context<'_>) -> Poll<WriteStatus>;
|
|
|
|
/// Gracefully shutdown filter
|
|
fn poll_shutdown(&self) -> Poll<sio::Result<()>>;
|
|
}
|
|
|
|
/// Creates new `Filter` values.
|
|
pub trait FilterFactory<F: Filter>: Sized {
|
|
/// The `Filter` value created by this factory
|
|
type Filter: Filter;
|
|
/// Errors produced while building a filter.
|
|
type Error: fmt::Debug;
|
|
/// The future of the `FilterFactory` instance.
|
|
type Future: Future<Output = Result<Io<Self::Filter>, Self::Error>>;
|
|
|
|
/// Create and return a new filter value asynchronously.
|
|
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>>;
|
|
}
|
|
|
|
/// Io status
|
|
#[derive(Debug)]
|
|
pub enum IoStatusUpdate {
|
|
/// Keep-alive timeout occured
|
|
KeepAlive,
|
|
/// Write backpressure is enabled
|
|
WriteBackpressure,
|
|
/// Stop io stream handling
|
|
Stop,
|
|
/// Peer is disconnected
|
|
PeerGone(Option<sio::Error>),
|
|
}
|
|
|
|
/// Recv error
|
|
#[derive(Debug)]
|
|
pub enum RecvError<U: Decoder> {
|
|
/// Keep-alive timeout occured
|
|
KeepAlive,
|
|
/// Write backpressure is enabled
|
|
WriteBackpressure,
|
|
/// Stop io stream handling
|
|
Stop,
|
|
/// 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)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#[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")
|
|
);
|
|
}
|
|
}
|