mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-03 21:07:39 +03:00
Add Framed type
This commit is contained in:
parent
7e3a4c2d00
commit
c801a9ea57
3 changed files with 126 additions and 0 deletions
|
@ -6,6 +6,8 @@
|
|||
|
||||
* Remove redundant Io::poll_write_backpressure() method.
|
||||
|
||||
* Add Framed type
|
||||
|
||||
* Fix read filters ordering
|
||||
|
||||
## [0.1.0-b.5] - 2021-12-24
|
||||
|
|
122
ntex-io/src/framed.rs
Normal file
122
ntex-io/src/framed.rs
Normal file
|
@ -0,0 +1,122 @@
|
|||
use std::{fmt, io};
|
||||
|
||||
use ntex_codec::{Decoder, Encoder};
|
||||
use ntex_util::future::Either;
|
||||
|
||||
use crate::Io;
|
||||
|
||||
/// A unified interface to an underlying I/O object, using
|
||||
/// the `Encoder` and `Decoder` traits to encode and decode frames.
|
||||
/// `Framed` is heavily optimized for streaming io.
|
||||
pub struct Framed<F, U> {
|
||||
io: Io<F>,
|
||||
codec: U,
|
||||
}
|
||||
|
||||
impl<F, U> Framed<F, U>
|
||||
where
|
||||
U: Decoder + Encoder,
|
||||
{
|
||||
#[inline]
|
||||
/// Provides an interface for reading and writing to
|
||||
/// `Io` object, using `Decode` and `Encode` traits of codec.
|
||||
pub fn new(io: Io<F>, codec: U) -> Framed<F, U> {
|
||||
Framed { io, codec }
|
||||
}
|
||||
|
||||
#[inline]
|
||||
/// Returns a reference to the underlying I/O stream wrapped by `Framed`.
|
||||
pub fn get_io(&self) -> &Io<F> {
|
||||
&self.io
|
||||
}
|
||||
|
||||
#[inline]
|
||||
/// Returns a reference to the underlying codec.
|
||||
pub fn get_codec(&self) -> &U {
|
||||
&self.codec
|
||||
}
|
||||
|
||||
#[inline]
|
||||
/// Wake write task and instruct to flush data.
|
||||
///
|
||||
/// This is async version of .poll_flush() method.
|
||||
pub async fn flush(&self, full: bool) -> Result<(), io::Error> {
|
||||
self.io.flush(full).await
|
||||
}
|
||||
|
||||
#[inline]
|
||||
/// Shut down io stream
|
||||
pub async fn shutdown(&self) -> Result<(), io::Error> {
|
||||
self.io.shutdown().await
|
||||
}
|
||||
}
|
||||
|
||||
impl<F, U> Framed<F, U>
|
||||
where
|
||||
U: Decoder,
|
||||
{
|
||||
#[inline]
|
||||
/// Read incoming io stream and decode codec item.
|
||||
pub async fn recv(&self) -> Result<Option<U::Item>, Either<U::Error, io::Error>> {
|
||||
self.io.recv(&self.codec).await
|
||||
}
|
||||
}
|
||||
|
||||
impl<F, U> fmt::Debug for Framed<F, U>
|
||||
where
|
||||
Io<F>: fmt::Debug,
|
||||
U: fmt::Debug,
|
||||
{
|
||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||
f.debug_struct("Framed")
|
||||
.field("io", &self.io)
|
||||
.field("codec", &self.codec)
|
||||
.finish()
|
||||
}
|
||||
}
|
||||
|
||||
impl<F, U> Framed<F, U>
|
||||
where
|
||||
U: Encoder,
|
||||
{
|
||||
#[inline]
|
||||
/// Serialize item and Write to the inner buffer
|
||||
pub async fn send(
|
||||
&self,
|
||||
item: <U as Encoder>::Item,
|
||||
) -> Result<(), Either<U::Error, io::Error>> {
|
||||
self.io.send(&self.codec, item).await
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use ntex_bytes::Bytes;
|
||||
use ntex_codec::BytesCodec;
|
||||
|
||||
use super::*;
|
||||
use crate::{testing::IoTest, Io};
|
||||
|
||||
#[ntex::test]
|
||||
async fn framed() {
|
||||
let (client, server) = IoTest::create();
|
||||
client.remote_buffer_cap(1024);
|
||||
client.write(b"chunk-0");
|
||||
|
||||
let server = Framed::new(Io::new(server), BytesCodec);
|
||||
server.get_codec();
|
||||
server.get_io();
|
||||
assert!(format!("{:?}", server).contains("Framed"));
|
||||
|
||||
let item = server.recv().await.unwrap().unwrap();
|
||||
assert_eq!(item, b"chunk-0".as_ref());
|
||||
|
||||
let data = Bytes::from_static(b"chunk-1");
|
||||
server.send(data).await.unwrap();
|
||||
server.flush(true).await.unwrap();
|
||||
assert_eq!(client.read_any(), b"chunk-1".as_ref());
|
||||
|
||||
server.shutdown().await.unwrap();
|
||||
assert!(client.is_closed());
|
||||
}
|
||||
}
|
|
@ -8,6 +8,7 @@ pub mod types;
|
|||
|
||||
mod dispatcher;
|
||||
mod filter;
|
||||
mod framed;
|
||||
mod io;
|
||||
mod ioref;
|
||||
mod seal;
|
||||
|
@ -26,6 +27,7 @@ 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};
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue