mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-04 13:27:39 +03:00
227 lines
6.3 KiB
Rust
227 lines
6.3 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;
|
|
pub mod utils;
|
|
|
|
#[cfg(feature = "async-std")]
|
|
mod asyncstd_rt;
|
|
#[cfg(any(feature = "tokio-traits", feature = "tokio"))]
|
|
mod tokio_impl;
|
|
#[cfg(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};
|
|
pub use self::seal::{IoBoxed, Sealed};
|
|
pub use self::tasks::{ReadContext, WriteContext};
|
|
pub use self::utils::filter;
|
|
|
|
/// 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 get_write_buf(&self) -> Option<BytesMut>;
|
|
|
|
fn release_read_buf(
|
|
&self,
|
|
io: &IoRef,
|
|
src: BytesMut,
|
|
dst: &mut Option<BytesMut>,
|
|
nbytes: usize,
|
|
) -> sio::Result<usize>;
|
|
|
|
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)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub mod rt {
|
|
//! async runtime helpers
|
|
|
|
#[cfg(feature = "tokio")]
|
|
pub use crate::tokio_rt::*;
|
|
|
|
#[cfg(all(not(feature = "tokio"), feature = "async-std"))]
|
|
pub use crate::asyncstd_rt::*;
|
|
|
|
#[cfg(all(not(feature = "tokio"), not(feature = "async-std")))]
|
|
pub fn spawn<F>(_: F) -> std::pin::Pin<Box<dyn std::future::Future<Output = F::Output>>>
|
|
where
|
|
F: std::future::Future + 'static,
|
|
{
|
|
unimplemented!()
|
|
}
|
|
}
|
|
|
|
#[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")
|
|
);
|
|
}
|
|
}
|