mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-04 05:17:39 +03:00
parent
e151b1eff1
commit
7960b550c9
84 changed files with 676 additions and 821 deletions
|
@ -26,7 +26,7 @@ ntex-io = { path = "ntex-io" }
|
||||||
ntex-http = { path = "ntex-http" }
|
ntex-http = { path = "ntex-http" }
|
||||||
ntex-router = { path = "ntex-router" }
|
ntex-router = { path = "ntex-router" }
|
||||||
ntex-rt = { path = "ntex-rt" }
|
ntex-rt = { path = "ntex-rt" }
|
||||||
#ntex-service = { path = "ntex-service" }
|
ntex-service = { path = "ntex-service" }
|
||||||
ntex-tls = { path = "ntex-tls" }
|
ntex-tls = { path = "ntex-tls" }
|
||||||
ntex-macros = { path = "ntex-macros" }
|
ntex-macros = { path = "ntex-macros" }
|
||||||
ntex-util = { path = "ntex-util" }
|
ntex-util = { path = "ntex-util" }
|
||||||
|
@ -34,3 +34,5 @@ ntex-util = { path = "ntex-util" }
|
||||||
ntex-glommio = { path = "ntex-glommio" }
|
ntex-glommio = { path = "ntex-glommio" }
|
||||||
ntex-tokio = { path = "ntex-tokio" }
|
ntex-tokio = { path = "ntex-tokio" }
|
||||||
ntex-async-std = { path = "ntex-async-std" }
|
ntex-async-std = { path = "ntex-async-std" }
|
||||||
|
|
||||||
|
ntex-h2 = { git = "https://github.com/ntex-rs/ntex-h2.git", branch = "0-3" }
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-async-std"
|
name = "ntex-async-std"
|
||||||
version = "0.2.2"
|
version = "0.3.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "async-std intergration for ntex framework"
|
description = "async-std intergration for ntex framework"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -17,8 +17,8 @@ path = "src/lib.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-bytes = "0.1.19"
|
ntex-bytes = "0.1.19"
|
||||||
ntex-io = "0.2.4"
|
ntex-io = "0.3.0"
|
||||||
ntex-util = "0.2.0"
|
ntex-util = "0.3.0"
|
||||||
async-oneshot = "0.5.0"
|
async-oneshot = "0.5.0"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
pin-project-lite = "0.2"
|
pin-project-lite = "0.2"
|
||||||
|
|
|
@ -27,4 +27,4 @@ simdutf8 = { version = "0.1.4", optional = true }
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
serde_test = "1.0"
|
serde_test = "1.0"
|
||||||
serde_json = "1.0"
|
serde_json = "1.0"
|
||||||
ntex = { version = "0.6.0", features = ["tokio"] }
|
ntex = { version = "0.7.0", features = ["tokio"] }
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-connect"
|
name = "ntex-connect"
|
||||||
version = "0.2.1"
|
version = "0.3.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "ntexwork connect utils for ntex framework"
|
description = "ntexwork connect utils for ntex framework"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -34,28 +34,28 @@ glommio = ["ntex-rt/glommio", "ntex-glommio"]
|
||||||
async-std = ["ntex-rt/async-std", "ntex-async-std"]
|
async-std = ["ntex-rt/async-std", "ntex-async-std"]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-service = "1.0.0"
|
ntex-service = "1.2.0"
|
||||||
ntex-bytes = "0.1.19"
|
ntex-bytes = "0.1.19"
|
||||||
ntex-http = "0.1.8"
|
ntex-http = "0.1.8"
|
||||||
ntex-io = "0.2.1"
|
ntex-io = "0.3.0"
|
||||||
ntex-rt = "0.4.7"
|
ntex-rt = "0.4.7"
|
||||||
ntex-tls = "0.2.1"
|
ntex-tls = "0.3.0"
|
||||||
ntex-util = "0.2.0"
|
ntex-util = "0.3.0"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
thiserror = "1.0"
|
thiserror = "1.0"
|
||||||
|
|
||||||
ntex-tokio = { version = "0.2.3", optional = true }
|
ntex-tokio = { version = "0.3.0", optional = true }
|
||||||
ntex-glommio = { version = "0.2.4", optional = true }
|
ntex-glommio = { version = "0.3.0", optional = true }
|
||||||
ntex-async-std = { version = "0.2.2", optional = true }
|
ntex-async-std = { version = "0.3.0", optional = true }
|
||||||
|
|
||||||
# openssl
|
# openssl
|
||||||
tls-openssl = { version="0.10", package = "openssl", optional = true }
|
tls-openssl = { version="0.10", package = "openssl", optional = true }
|
||||||
|
|
||||||
# rustls
|
# rustls
|
||||||
tls-rustls = { version = "0.20", package = "rustls", optional = true }
|
tls-rustls = { version = "0.21", package = "rustls", optional = true }
|
||||||
webpki-roots = { version = "0.22", optional = true }
|
webpki-roots = { version = "0.23", optional = true }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
rand = "0.8"
|
rand = "0.8"
|
||||||
env_logger = "0.10"
|
env_logger = "0.10"
|
||||||
ntex = { version = "0.6.0", features = ["tokio"] }
|
ntex = { version = "0.7.0", features = ["tokio"] }
|
||||||
|
|
|
@ -20,7 +20,6 @@ pub use self::resolve::Resolver;
|
||||||
pub use self::service::Connector;
|
pub use self::service::Connector;
|
||||||
|
|
||||||
use ntex_io::Io;
|
use ntex_io::Io;
|
||||||
use ntex_service::Service;
|
|
||||||
|
|
||||||
/// Resolve and connect to remote host
|
/// Resolve and connect to remote host
|
||||||
pub async fn connect<T, U>(message: U) -> Result<Io, ConnectError>
|
pub async fn connect<T, U>(message: U) -> Result<Io, ConnectError>
|
||||||
|
@ -28,7 +27,8 @@ where
|
||||||
T: Address,
|
T: Address,
|
||||||
Connect<T>: From<U>,
|
Connect<T>: From<U>,
|
||||||
{
|
{
|
||||||
service::ConnectServiceResponse::new(Resolver::new().call(message.into())).await
|
service::ConnectServiceResponse::new(Box::pin(Resolver::new().lookup(message.into())))
|
||||||
|
.await
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(unused_imports)]
|
#[allow(unused_imports)]
|
||||||
|
|
|
@ -5,22 +5,22 @@ pub use tls_openssl::ssl::{Error as SslError, HandshakeError, SslConnector, SslM
|
||||||
|
|
||||||
use ntex_bytes::PoolId;
|
use ntex_bytes::PoolId;
|
||||||
use ntex_io::{FilterFactory, Io, Layer};
|
use ntex_io::{FilterFactory, Io, Layer};
|
||||||
use ntex_service::{Service, ServiceFactory};
|
use ntex_service::{Container, Ctx, Service, ServiceFactory};
|
||||||
use ntex_tls::openssl::SslConnector as IoSslConnector;
|
use ntex_tls::openssl::SslConnector as IoSslConnector;
|
||||||
use ntex_util::future::{BoxFuture, Ready};
|
use ntex_util::future::{BoxFuture, Ready};
|
||||||
|
|
||||||
use super::{Address, Connect, ConnectError, Connector as BaseConnector};
|
use super::{Address, Connect, ConnectError, Connector as BaseConnector};
|
||||||
|
|
||||||
pub struct Connector<T> {
|
pub struct Connector<T> {
|
||||||
connector: BaseConnector<T>,
|
connector: Container<BaseConnector<T>>,
|
||||||
openssl: SslConnector,
|
openssl: SslConnector,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> Connector<T> {
|
impl<T: Address> Connector<T> {
|
||||||
/// Construct new OpensslConnectService factory
|
/// Construct new OpensslConnectService factory
|
||||||
pub fn new(connector: SslConnector) -> Self {
|
pub fn new(connector: SslConnector) -> Self {
|
||||||
Connector {
|
Connector {
|
||||||
connector: BaseConnector::default(),
|
connector: BaseConnector::default().into(),
|
||||||
openssl: connector,
|
openssl: connector,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -30,8 +30,15 @@ impl<T> Connector<T> {
|
||||||
/// Use specified memory pool for memory allocations. By default P0
|
/// Use specified memory pool for memory allocations. By default P0
|
||||||
/// memory pool is used.
|
/// memory pool is used.
|
||||||
pub fn memory_pool(self, id: PoolId) -> Self {
|
pub fn memory_pool(self, id: PoolId) -> Self {
|
||||||
|
let connector = self
|
||||||
|
.connector
|
||||||
|
.into_service()
|
||||||
|
.expect("Connector has been cloned")
|
||||||
|
.memory_pool(id)
|
||||||
|
.into();
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
connector: self.connector.memory_pool(id),
|
connector,
|
||||||
openssl: self.openssl,
|
openssl: self.openssl,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -100,7 +107,7 @@ impl<T: Address> Service<Connect<T>> for Connector<T> {
|
||||||
type Future<'f> = BoxFuture<'f, Result<Self::Response, Self::Error>>;
|
type Future<'f> = BoxFuture<'f, Result<Self::Response, Self::Error>>;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: Connect<T>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: Connect<T>, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
Box::pin(self.connect(req))
|
Box::pin(self.connect(req))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -108,7 +115,7 @@ impl<T: Address> Service<Connect<T>> for Connector<T> {
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
use ntex_service::{Service, ServiceFactory};
|
use ntex_service::ServiceFactory;
|
||||||
|
|
||||||
#[ntex::test]
|
#[ntex::test]
|
||||||
async fn test_openssl_connect() {
|
async fn test_openssl_connect() {
|
||||||
|
@ -117,9 +124,9 @@ mod tests {
|
||||||
});
|
});
|
||||||
|
|
||||||
let ssl = SslConnector::builder(SslMethod::tls()).unwrap();
|
let ssl = SslConnector::builder(SslMethod::tls()).unwrap();
|
||||||
let factory = Connector::new(ssl.build()).clone().memory_pool(PoolId::P5);
|
let factory = Connector::new(ssl.build()).memory_pool(PoolId::P5).clone();
|
||||||
|
|
||||||
let srv = factory.create(&()).await.unwrap();
|
let srv = factory.container(&()).await.unwrap();
|
||||||
let result = srv
|
let result = srv
|
||||||
.call(Connect::new("").set_addr(Some(server.addr())))
|
.call(Connect::new("").set_addr(Some(server.addr())))
|
||||||
.await;
|
.await;
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use std::{fmt, io, marker, net};
|
use std::{fmt, io, marker, net};
|
||||||
|
|
||||||
use ntex_rt::spawn_blocking;
|
use ntex_rt::spawn_blocking;
|
||||||
use ntex_service::{Service, ServiceFactory};
|
use ntex_service::{Ctx, Service, ServiceFactory};
|
||||||
use ntex_util::future::{BoxFuture, Either, Ready};
|
use ntex_util::future::{BoxFuture, Either, Ready};
|
||||||
|
|
||||||
use crate::{Address, Connect, ConnectError};
|
use crate::{Address, Connect, ConnectError};
|
||||||
|
@ -115,7 +115,7 @@ impl<T: Address> Service<Connect<T>> for Resolver<T> {
|
||||||
type Future<'f> = BoxFuture<'f, Result<Connect<T>, Self::Error>>;
|
type Future<'f> = BoxFuture<'f, Result<Connect<T>, Self::Error>>;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: Connect<T>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: Connect<T>, _: Ctx<'a, Self>) -> Self::Future<'_> {
|
||||||
Box::pin(self.lookup(req))
|
Box::pin(self.lookup(req))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -129,7 +129,7 @@ mod tests {
|
||||||
async fn resolver() {
|
async fn resolver() {
|
||||||
let resolver = Resolver::default().clone();
|
let resolver = Resolver::default().clone();
|
||||||
assert!(format!("{:?}", resolver).contains("Resolver"));
|
assert!(format!("{:?}", resolver).contains("Resolver"));
|
||||||
let srv = resolver.create(()).await.unwrap();
|
let srv = resolver.container(()).await.unwrap();
|
||||||
assert!(lazy(|cx| srv.poll_ready(cx)).await.is_ready());
|
assert!(lazy(|cx| srv.poll_ready(cx)).await.is_ready());
|
||||||
|
|
||||||
let res = srv.call(Connect::new("www.rust-lang.org")).await;
|
let res = srv.call(Connect::new("www.rust-lang.org")).await;
|
||||||
|
|
|
@ -5,7 +5,7 @@ pub use tls_rustls::{ClientConfig, ServerName};
|
||||||
|
|
||||||
use ntex_bytes::PoolId;
|
use ntex_bytes::PoolId;
|
||||||
use ntex_io::{FilterFactory, Io, Layer};
|
use ntex_io::{FilterFactory, Io, Layer};
|
||||||
use ntex_service::{Service, ServiceFactory};
|
use ntex_service::{Container, Ctx, Service, ServiceFactory};
|
||||||
use ntex_tls::rustls::TlsConnector;
|
use ntex_tls::rustls::TlsConnector;
|
||||||
use ntex_util::future::{BoxFuture, Ready};
|
use ntex_util::future::{BoxFuture, Ready};
|
||||||
|
|
||||||
|
@ -13,24 +13,24 @@ use super::{Address, Connect, ConnectError, Connector as BaseConnector};
|
||||||
|
|
||||||
/// Rustls connector factory
|
/// Rustls connector factory
|
||||||
pub struct Connector<T> {
|
pub struct Connector<T> {
|
||||||
connector: BaseConnector<T>,
|
connector: Container<BaseConnector<T>>,
|
||||||
inner: TlsConnector,
|
inner: TlsConnector,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> From<std::sync::Arc<ClientConfig>> for Connector<T> {
|
impl<T: Address> From<std::sync::Arc<ClientConfig>> for Connector<T> {
|
||||||
fn from(cfg: std::sync::Arc<ClientConfig>) -> Self {
|
fn from(cfg: std::sync::Arc<ClientConfig>) -> Self {
|
||||||
Connector {
|
Connector {
|
||||||
inner: TlsConnector::new(cfg),
|
inner: TlsConnector::new(cfg),
|
||||||
connector: BaseConnector::default(),
|
connector: BaseConnector::default().into(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> Connector<T> {
|
impl<T: Address> Connector<T> {
|
||||||
pub fn new(config: ClientConfig) -> Self {
|
pub fn new(config: ClientConfig) -> Self {
|
||||||
Connector {
|
Connector {
|
||||||
inner: TlsConnector::new(std::sync::Arc::new(config)),
|
inner: TlsConnector::new(std::sync::Arc::new(config)),
|
||||||
connector: BaseConnector::default(),
|
connector: BaseConnector::default().into(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,8 +39,14 @@ impl<T> Connector<T> {
|
||||||
/// Use specified memory pool for memory allocations. By default P0
|
/// Use specified memory pool for memory allocations. By default P0
|
||||||
/// memory pool is used.
|
/// memory pool is used.
|
||||||
pub fn memory_pool(self, id: PoolId) -> Self {
|
pub fn memory_pool(self, id: PoolId) -> Self {
|
||||||
|
let connector = self
|
||||||
|
.connector
|
||||||
|
.into_service()
|
||||||
|
.unwrap()
|
||||||
|
.memory_pool(id)
|
||||||
|
.into();
|
||||||
Self {
|
Self {
|
||||||
connector: self.connector.memory_pool(id),
|
connector,
|
||||||
inner: self.inner,
|
inner: self.inner,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -104,7 +110,7 @@ impl<T: Address> Service<Connect<T>> for Connector<T> {
|
||||||
type Error = ConnectError;
|
type Error = ConnectError;
|
||||||
type Future<'f> = BoxFuture<'f, Result<Self::Response, Self::Error>>;
|
type Future<'f> = BoxFuture<'f, Result<Self::Response, Self::Error>>;
|
||||||
|
|
||||||
fn call(&self, req: Connect<T>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: Connect<T>, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
Box::pin(self.connect(req))
|
Box::pin(self.connect(req))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -115,7 +121,7 @@ mod tests {
|
||||||
use tls_rustls::{OwnedTrustAnchor, RootCertStore};
|
use tls_rustls::{OwnedTrustAnchor, RootCertStore};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use ntex_service::{Service, ServiceFactory};
|
use ntex_service::ServiceFactory;
|
||||||
use ntex_util::future::lazy;
|
use ntex_util::future::lazy;
|
||||||
|
|
||||||
#[ntex::test]
|
#[ntex::test]
|
||||||
|
@ -139,9 +145,11 @@ mod tests {
|
||||||
.with_root_certificates(cert_store)
|
.with_root_certificates(cert_store)
|
||||||
.with_no_client_auth();
|
.with_no_client_auth();
|
||||||
let _ = Connector::<&'static str>::new(config.clone()).clone();
|
let _ = Connector::<&'static str>::new(config.clone()).clone();
|
||||||
let factory = Connector::from(Arc::new(config)).memory_pool(PoolId::P5);
|
let factory = Connector::from(Arc::new(config))
|
||||||
|
.memory_pool(PoolId::P5)
|
||||||
|
.clone();
|
||||||
|
|
||||||
let srv = factory.create(&()).await.unwrap();
|
let srv = factory.container(&()).await.unwrap();
|
||||||
// always ready
|
// always ready
|
||||||
assert!(lazy(|cx| srv.poll_ready(cx)).await.is_ready());
|
assert!(lazy(|cx| srv.poll_ready(cx)).await.is_ready());
|
||||||
let result = srv
|
let result = srv
|
||||||
|
|
|
@ -3,7 +3,7 @@ use std::{collections::VecDeque, future::Future, io, net::SocketAddr, pin::Pin};
|
||||||
|
|
||||||
use ntex_bytes::{PoolId, PoolRef};
|
use ntex_bytes::{PoolId, PoolRef};
|
||||||
use ntex_io::{types, Io};
|
use ntex_io::{types, Io};
|
||||||
use ntex_service::{Service, ServiceFactory};
|
use ntex_service::{Ctx, Service, ServiceFactory};
|
||||||
use ntex_util::future::{BoxFuture, Either, Ready};
|
use ntex_util::future::{BoxFuture, Either, Ready};
|
||||||
|
|
||||||
use crate::{net::tcp_connect_in, Address, Connect, ConnectError, Resolver};
|
use crate::{net::tcp_connect_in, Address, Connect, ConnectError, Resolver};
|
||||||
|
@ -39,7 +39,7 @@ impl<T: Address> Connector<T> {
|
||||||
Connect<T>: From<U>,
|
Connect<T>: From<U>,
|
||||||
{
|
{
|
||||||
ConnectServiceResponse {
|
ConnectServiceResponse {
|
||||||
state: ConnectState::Resolve(self.resolver.call(message.into())),
|
state: ConnectState::Resolve(Box::pin(self.resolver.lookup(message.into()))),
|
||||||
pool: self.pool,
|
pool: self.pool,
|
||||||
}
|
}
|
||||||
.await
|
.await
|
||||||
|
@ -80,13 +80,13 @@ impl<T: Address> Service<Connect<T>> for Connector<T> {
|
||||||
type Future<'f> = ConnectServiceResponse<'f, T>;
|
type Future<'f> = ConnectServiceResponse<'f, T>;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: Connect<T>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: Connect<T>, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
ConnectServiceResponse::new(self.resolver.call(req))
|
ConnectServiceResponse::new(Box::pin(self.resolver.lookup(req)))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
enum ConnectState<'f, T: Address> {
|
enum ConnectState<'f, T: Address> {
|
||||||
Resolve(<Resolver<T> as Service<Connect<T>>>::Future<'f>),
|
Resolve(BoxFuture<'f, Result<Connect<T>, ConnectError>>),
|
||||||
Connect(TcpConnectorResponse<T>),
|
Connect(TcpConnectorResponse<T>),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -97,7 +97,7 @@ pub struct ConnectServiceResponse<'f, T: Address> {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'f, T: Address> ConnectServiceResponse<'f, T> {
|
impl<'f, T: Address> ConnectServiceResponse<'f, T> {
|
||||||
pub(super) fn new(fut: <Resolver<T> as Service<Connect<T>>>::Future<'f>) -> Self {
|
pub(super) fn new(fut: BoxFuture<'f, Result<Connect<T>, ConnectError>>) -> Self {
|
||||||
Self {
|
Self {
|
||||||
state: ConnectState::Resolve(fut),
|
state: ConnectState::Resolve(fut),
|
||||||
pool: PoolId::P0.pool_ref(),
|
pool: PoolId::P0.pool_ref(),
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-glommio"
|
name = "ntex-glommio"
|
||||||
version = "0.2.4"
|
version = "0.3.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "glommio intergration for ntex framework"
|
description = "glommio intergration for ntex framework"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -17,8 +17,8 @@ path = "src/lib.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-bytes = "0.1.19"
|
ntex-bytes = "0.1.19"
|
||||||
ntex-io = "0.2.4"
|
ntex-io = "0.3.0"
|
||||||
ntex-util = "0.2.0"
|
ntex-util = "0.3.0"
|
||||||
async-oneshot = "0.5.0"
|
async-oneshot = "0.5.0"
|
||||||
futures-lite = "1.12"
|
futures-lite = "1.12"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
#![allow(
|
#![allow(
|
||||||
clippy::derive_hash_xor_eq,
|
|
||||||
clippy::should_implement_trait,
|
clippy::should_implement_trait,
|
||||||
clippy::no_effect,
|
clippy::no_effect,
|
||||||
clippy::missing_safety_doc
|
clippy::missing_safety_doc
|
||||||
|
@ -8,6 +7,7 @@ use std::{cmp, error::Error, fmt, str, str::FromStr};
|
||||||
|
|
||||||
use ntex_bytes::{ByteString, Bytes};
|
use ntex_bytes::{ByteString, Bytes};
|
||||||
|
|
||||||
|
#[allow(clippy::derived_hash_with_manual_eq)]
|
||||||
/// Represents an HTTP header field value.
|
/// Represents an HTTP header field value.
|
||||||
///
|
///
|
||||||
/// In practice, HTTP header field values are usually valid ASCII. However, the
|
/// In practice, HTTP header field values are usually valid ASCII. However, the
|
||||||
|
@ -17,7 +17,7 @@ use ntex_bytes::{ByteString, Bytes};
|
||||||
/// To handle this, the `HeaderValue` is useable as a type and can be compared
|
/// To handle this, the `HeaderValue` is useable as a type and can be compared
|
||||||
/// with strings and implements `Debug`. A `to_str` fn is provided that returns
|
/// with strings and implements `Debug`. A `to_str` fn is provided that returns
|
||||||
/// an `Err` if the header value contains non visible ascii characters.
|
/// an `Err` if the header value contains non visible ascii characters.
|
||||||
#[derive(Clone, Hash)]
|
#[derive(Clone, Hash, Eq)]
|
||||||
pub struct HeaderValue {
|
pub struct HeaderValue {
|
||||||
inner: Bytes,
|
inner: Bytes,
|
||||||
is_sensitive: bool,
|
is_sensitive: bool,
|
||||||
|
@ -542,8 +542,6 @@ impl PartialEq for HeaderValue {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Eq for HeaderValue {}
|
|
||||||
|
|
||||||
impl PartialOrd for HeaderValue {
|
impl PartialOrd for HeaderValue {
|
||||||
#[inline]
|
#[inline]
|
||||||
fn partial_cmp(&self, other: &HeaderValue) -> Option<cmp::Ordering> {
|
fn partial_cmp(&self, other: &HeaderValue) -> Option<cmp::Ordering> {
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-io"
|
name = "ntex-io"
|
||||||
version = "0.2.10"
|
version = "0.3.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "Utilities for encoding and decoding frames"
|
description = "Utilities for encoding and decoding frames"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -18,8 +18,8 @@ path = "src/lib.rs"
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-codec = "0.6.2"
|
ntex-codec = "0.6.2"
|
||||||
ntex-bytes = "0.1.19"
|
ntex-bytes = "0.1.19"
|
||||||
ntex-util = "0.2.2"
|
ntex-util = "0.3.0"
|
||||||
ntex-service = "1.0.2"
|
ntex-service = "1.2.0"
|
||||||
|
|
||||||
bitflags = "1.3"
|
bitflags = "1.3"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
|
@ -29,4 +29,4 @@ pin-project-lite = "0.2"
|
||||||
rand = "0.8"
|
rand = "0.8"
|
||||||
env_logger = "0.10"
|
env_logger = "0.10"
|
||||||
|
|
||||||
ntex = { version = "0.6.3", features = ["tokio"] }
|
ntex = { version = "0.7.0", features = ["tokio"] }
|
||||||
|
|
|
@ -376,14 +376,12 @@ impl<'a> ReadBuf<'a> {
|
||||||
if let Some(dst) = dst {
|
if let Some(dst) = dst {
|
||||||
if dst.is_empty() {
|
if dst.is_empty() {
|
||||||
self.io.memory_pool().release_read_buf(dst);
|
self.io.memory_pool().release_read_buf(dst);
|
||||||
|
} else if let Some(mut buf) = self.curr.0.take() {
|
||||||
|
buf.extend_from_slice(&dst);
|
||||||
|
self.curr.0.set(Some(buf));
|
||||||
|
self.io.memory_pool().release_read_buf(dst);
|
||||||
} else {
|
} else {
|
||||||
if let Some(mut buf) = self.curr.0.take() {
|
self.curr.0.set(Some(dst));
|
||||||
buf.extend_from_slice(&dst);
|
|
||||||
self.curr.0.set(Some(buf));
|
|
||||||
self.io.memory_pool().release_read_buf(dst);
|
|
||||||
} else {
|
|
||||||
self.curr.0.set(Some(dst));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,7 +3,7 @@ use std::{cell::Cell, future, pin::Pin, rc::Rc, task::Context, task::Poll, time}
|
||||||
|
|
||||||
use ntex_bytes::Pool;
|
use ntex_bytes::Pool;
|
||||||
use ntex_codec::{Decoder, Encoder};
|
use ntex_codec::{Decoder, Encoder};
|
||||||
use ntex_service::{IntoService, Service};
|
use ntex_service::{Container, IntoService, Service};
|
||||||
use ntex_util::time::Seconds;
|
use ntex_util::time::Seconds;
|
||||||
use ntex_util::{future::Either, ready, spawn};
|
use ntex_util::{future::Either, ready, spawn};
|
||||||
|
|
||||||
|
@ -51,7 +51,7 @@ where
|
||||||
{
|
{
|
||||||
io: IoBoxed,
|
io: IoBoxed,
|
||||||
codec: U,
|
codec: U,
|
||||||
service: S,
|
service: Container<S>,
|
||||||
error: Cell<Option<DispatcherError<S::Error, <U as Encoder>::Error>>>,
|
error: Cell<Option<DispatcherError<S::Error, <U as Encoder>::Error>>>,
|
||||||
inflight: Cell<usize>,
|
inflight: Cell<usize>,
|
||||||
}
|
}
|
||||||
|
@ -107,7 +107,7 @@ where
|
||||||
codec,
|
codec,
|
||||||
error: Cell::new(None),
|
error: Cell::new(None),
|
||||||
inflight: Cell::new(0),
|
inflight: Cell::new(0),
|
||||||
service: service.into_service(),
|
service: Container::new(service.into_service()),
|
||||||
});
|
});
|
||||||
|
|
||||||
Dispatcher {
|
Dispatcher {
|
||||||
|
@ -340,7 +340,7 @@ where
|
||||||
{
|
{
|
||||||
fn poll_service(
|
fn poll_service(
|
||||||
&self,
|
&self,
|
||||||
srv: &S,
|
srv: &Container<S>,
|
||||||
cx: &mut Context<'_>,
|
cx: &mut Context<'_>,
|
||||||
io: &IoBoxed,
|
io: &IoBoxed,
|
||||||
) -> Poll<PollService<U>> {
|
) -> Poll<PollService<U>> {
|
||||||
|
@ -426,6 +426,7 @@ mod tests {
|
||||||
|
|
||||||
use ntex_bytes::{Bytes, PoolId, PoolRef};
|
use ntex_bytes::{Bytes, PoolId, PoolRef};
|
||||||
use ntex_codec::BytesCodec;
|
use ntex_codec::BytesCodec;
|
||||||
|
use ntex_service::Ctx;
|
||||||
use ntex_util::{future::Ready, time::sleep, time::Millis, time::Seconds};
|
use ntex_util::{future::Ready, time::sleep, time::Millis, time::Seconds};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
|
@ -475,7 +476,7 @@ mod tests {
|
||||||
io: state.into(),
|
io: state.into(),
|
||||||
error: Cell::new(None),
|
error: Cell::new(None),
|
||||||
inflight: Cell::new(0),
|
inflight: Cell::new(0),
|
||||||
service: service.into_service(),
|
service: Container::new(service.into_service()),
|
||||||
});
|
});
|
||||||
|
|
||||||
(
|
(
|
||||||
|
@ -621,7 +622,11 @@ mod tests {
|
||||||
Poll::Ready(Err(()))
|
Poll::Ready(Err(()))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, _: DispatchItem<BytesCodec>) -> Self::Future<'_> {
|
fn call<'a>(
|
||||||
|
&'a self,
|
||||||
|
_: DispatchItem<BytesCodec>,
|
||||||
|
_: Ctx<'a, Self>,
|
||||||
|
) -> Self::Future<'a> {
|
||||||
Ready::Ok(None)
|
Ready::Ok(None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -80,20 +80,15 @@ impl Default for IoTestFlags {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug, Default)]
|
||||||
enum IoTestState {
|
enum IoTestState {
|
||||||
|
#[default]
|
||||||
Ok,
|
Ok,
|
||||||
Pending,
|
Pending,
|
||||||
Close,
|
Close,
|
||||||
Err(io::Error),
|
Err(io::Error),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for IoTestState {
|
|
||||||
fn default() -> Self {
|
|
||||||
IoTestState::Ok
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl IoTest {
|
impl IoTest {
|
||||||
/// Create a two interconnected streams
|
/// Create a two interconnected streams
|
||||||
pub fn create() -> (IoTest, IoTest) {
|
pub fn create() -> (IoTest, IoTest) {
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
|
|
||||||
use ntex_service::{fn_service, pipeline_factory, Service, ServiceFactory};
|
use ntex_service::{fn_service, pipeline_factory, Ctx, Service, ServiceFactory};
|
||||||
use ntex_util::future::Ready;
|
use ntex_util::future::Ready;
|
||||||
|
|
||||||
use crate::{Filter, FilterFactory, Io, IoBoxed, Layer};
|
use crate::{Filter, FilterFactory, Io, IoBoxed, Layer};
|
||||||
|
@ -75,7 +75,7 @@ where
|
||||||
type Future<'f> = T::Future where T: 'f, F: 'f;
|
type Future<'f> = T::Future where T: 'f, F: 'f;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: Io<F>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: Io<F>, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
self.filter.clone().create(req)
|
self.filter.clone().create(req)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -106,7 +106,7 @@ mod tests {
|
||||||
.unwrap();
|
.unwrap();
|
||||||
Ok::<_, ()>(())
|
Ok::<_, ()>(())
|
||||||
}))
|
}))
|
||||||
.create(())
|
.container(())
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
let _ = svc.call(Io::new(server)).await;
|
let _ = svc.call(Io::new(server)).await;
|
||||||
|
@ -152,7 +152,7 @@ mod tests {
|
||||||
let _ = io.recv(&BytesCodec).await;
|
let _ = io.recv(&BytesCodec).await;
|
||||||
Ok::<_, ()>(())
|
Ok::<_, ()>(())
|
||||||
})))
|
})))
|
||||||
.create(())
|
.container(())
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
let _ = svc.call(Io::new(server)).await;
|
let _ = svc.call(Io::new(server)).await;
|
||||||
|
|
|
@ -16,6 +16,6 @@ syn = { version = "^1", features = ["full", "parsing"] }
|
||||||
proc-macro2 = "^1"
|
proc-macro2 = "^1"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
ntex = { version = "0.6.0-alpha.0", features = ["tokio"] }
|
ntex = { version = "0.7.0", features = ["tokio"] }
|
||||||
futures = "0.3"
|
futures = "0.3"
|
||||||
env_logger = "0.10"
|
env_logger = "0.10"
|
|
@ -50,7 +50,7 @@ pub(super) fn requote(val: &[u8]) -> Option<String> {
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn from_hex(v: u8) -> Option<u8> {
|
fn from_hex(v: u8) -> Option<u8> {
|
||||||
if (b'0'..=b'9').contains(&v) {
|
if v.is_ascii_digit() {
|
||||||
Some(v - 0x30) // ord('0') == 0x30
|
Some(v - 0x30) // ord('0') == 0x30
|
||||||
} else if (b'A'..=b'F').contains(&v) {
|
} else if (b'A'..=b'F').contains(&v) {
|
||||||
Some(v - 0x41 + 10) // ord('A') == 0x41
|
Some(v - 0x41 + 10) // ord('A') == 0x41
|
||||||
|
|
|
@ -20,5 +20,5 @@ pin-project-lite = "0.2.6"
|
||||||
slab = "0.4"
|
slab = "0.4"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
ntex = { version = "0.6.0", features = ["tokio"] }
|
ntex = { version = "0.7.0", features = ["tokio"] }
|
||||||
ntex-util = "0.2.0"
|
ntex-util = "0.3.0"
|
||||||
|
|
|
@ -59,10 +59,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'a>(&'a self, req: Req, ctx: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: Req, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
Req: 'a,
|
|
||||||
{
|
|
||||||
AndThenServiceResponse {
|
AndThenServiceResponse {
|
||||||
slf: self,
|
slf: self,
|
||||||
state: State::A {
|
state: State::A {
|
||||||
|
@ -253,10 +250,7 @@ mod tests {
|
||||||
Poll::Ready(Ok(()))
|
Poll::Ready(Ok(()))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call<'a>(&'a self, req: &'static str, _: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: &'static str, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
&'static str: 'a,
|
|
||||||
{
|
|
||||||
Ready::Ok(req)
|
Ready::Ok(req)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -274,10 +268,7 @@ mod tests {
|
||||||
Poll::Ready(Ok(()))
|
Poll::Ready(Ok(()))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call<'a>(&'a self, req: &'static str, _: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: &'static str, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
&'static str: 'a,
|
|
||||||
{
|
|
||||||
Ready::Ok((req, "srv2"))
|
Ready::Ok((req, "srv2"))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -101,10 +101,7 @@ where
|
||||||
crate::forward_poll_shutdown!(service);
|
crate::forward_poll_shutdown!(service);
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'a>(&'a self, req: In, ctx: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: In, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
In: 'a,
|
|
||||||
{
|
|
||||||
let (index, waiters) = ctx.into_inner();
|
let (index, waiters) = ctx.into_inner();
|
||||||
let svc = ApplyService {
|
let svc = ApplyService {
|
||||||
index,
|
index,
|
||||||
|
@ -235,10 +232,7 @@ mod tests {
|
||||||
type Error = ();
|
type Error = ();
|
||||||
type Future<'f> = Ready<(), ()>;
|
type Future<'f> = Ready<(), ()>;
|
||||||
|
|
||||||
fn call<'a>(&'a self, _: (), _: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, _: (), _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
(): 'a,
|
|
||||||
{
|
|
||||||
Ready::Ok(())
|
Ready::Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -62,9 +62,7 @@ pub trait ServiceObj<Req> {
|
||||||
req: Req,
|
req: Req,
|
||||||
idx: usize,
|
idx: usize,
|
||||||
waiters: &'a Rc<RefCell<slab::Slab<Option<Waker>>>>,
|
waiters: &'a Rc<RefCell<slab::Slab<Option<Waker>>>>,
|
||||||
) -> BoxFuture<'a, Self::Response, Self::Error>
|
) -> BoxFuture<'a, Self::Response, Self::Error>;
|
||||||
where
|
|
||||||
Req: 'a;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S, Req> ServiceObj<Req> for S
|
impl<S, Req> ServiceObj<Req> for S
|
||||||
|
@ -91,10 +89,7 @@ where
|
||||||
req: Req,
|
req: Req,
|
||||||
idx: usize,
|
idx: usize,
|
||||||
waiters: &'a Rc<RefCell<slab::Slab<Option<Waker>>>>,
|
waiters: &'a Rc<RefCell<slab::Slab<Option<Waker>>>>,
|
||||||
) -> BoxFuture<'a, Self::Response, Self::Error>
|
) -> BoxFuture<'a, Self::Response, Self::Error> {
|
||||||
where
|
|
||||||
Req: 'a,
|
|
||||||
{
|
|
||||||
Box::pin(Ctx::<'a, S>::new(idx, waiters).call_nowait(self, req))
|
Box::pin(Ctx::<'a, S>::new(idx, waiters).call_nowait(self, req))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -194,10 +189,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'a>(&'a self, req: Req, ctx: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: Req, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
Req: 'a,
|
|
||||||
{
|
|
||||||
let (index, waiters) = ctx.into_inner();
|
let (index, waiters) = ctx.into_inner();
|
||||||
self.0.call(req, index, waiters)
|
self.0.call(req, index, waiters)
|
||||||
}
|
}
|
||||||
|
@ -230,10 +222,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'a>(&'a self, req: Req, ctx: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: Req, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
Req: 'a,
|
|
||||||
{
|
|
||||||
let (index, waiters) = ctx.into_inner();
|
let (index, waiters) = ctx.into_inner();
|
||||||
self.0.call(req, index, waiters)
|
self.0.call(req, index, waiters)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,18 +1,14 @@
|
||||||
use std::{cell::RefCell, future::Future, marker, pin::Pin, rc::Rc, task, task::Poll};
|
use std::{cell::RefCell, future::Future, marker, ops, pin::Pin, rc::Rc, task, task::Poll};
|
||||||
|
|
||||||
use crate::{Service, ServiceFactory};
|
use crate::{Service, ServiceFactory};
|
||||||
|
|
||||||
pub struct Container<S, R> {
|
pub struct Container<S> {
|
||||||
svc: Rc<S>,
|
svc: Rc<S>,
|
||||||
index: usize,
|
index: usize,
|
||||||
waiters: Rc<RefCell<slab::Slab<Option<task::Waker>>>>,
|
waiters: Rc<RefCell<slab::Slab<Option<task::Waker>>>>,
|
||||||
_t: marker::PhantomData<R>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S, R> Container<S, R>
|
impl<S> Container<S> {
|
||||||
where
|
|
||||||
S: Service<R>,
|
|
||||||
{
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn new(svc: S) -> Self {
|
pub fn new(svc: S) -> Self {
|
||||||
let mut waiters = slab::Slab::new();
|
let mut waiters = slab::Slab::new();
|
||||||
|
@ -21,13 +17,15 @@ where
|
||||||
index,
|
index,
|
||||||
svc: Rc::new(svc),
|
svc: Rc::new(svc),
|
||||||
waiters: Rc::new(RefCell::new(waiters)),
|
waiters: Rc::new(RefCell::new(waiters)),
|
||||||
_t: marker::PhantomData,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
/// Returns `Ready` when the service is able to process requests.
|
/// Returns `Ready` when the service is able to process requests.
|
||||||
pub fn poll_ready(&self, cx: &mut task::Context<'_>) -> Poll<Result<(), S::Error>> {
|
pub fn poll_ready<R>(&self, cx: &mut task::Context<'_>) -> Poll<Result<(), S::Error>>
|
||||||
|
where
|
||||||
|
S: Service<R>,
|
||||||
|
{
|
||||||
let res = self.svc.poll_ready(cx);
|
let res = self.svc.poll_ready(cx);
|
||||||
|
|
||||||
if res.is_pending() {
|
if res.is_pending() {
|
||||||
|
@ -38,13 +36,19 @@ where
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
/// Shutdown enclosed service.
|
/// Shutdown enclosed service.
|
||||||
pub fn poll_shutdown(&self, cx: &mut task::Context<'_>) -> Poll<()> {
|
pub fn poll_shutdown<R>(&self, cx: &mut task::Context<'_>) -> Poll<()>
|
||||||
|
where
|
||||||
|
S: Service<R>,
|
||||||
|
{
|
||||||
self.svc.poll_shutdown(cx)
|
self.svc.poll_shutdown(cx)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
/// Process the request and return the response asynchronously.
|
/// Process the request and return the response asynchronously.
|
||||||
pub fn call<'a>(&'a self, req: R) -> ServiceCall<'a, S, R> {
|
pub fn call<'a, R>(&'a self, req: R) -> ServiceCall<'a, S, R>
|
||||||
|
where
|
||||||
|
S: Service<R>,
|
||||||
|
{
|
||||||
let ctx = Ctx::<'a, S> {
|
let ctx = Ctx::<'a, S> {
|
||||||
index: self.index,
|
index: self.index,
|
||||||
waiters: &self.waiters,
|
waiters: &self.waiters,
|
||||||
|
@ -53,7 +57,7 @@ where
|
||||||
ctx.call(self.svc.as_ref(), req)
|
ctx.call(self.svc.as_ref(), req)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn create<F: ServiceFactory<R, C>, C>(
|
pub(crate) fn create<F: ServiceFactory<R, C>, R, C>(
|
||||||
f: &F,
|
f: &F,
|
||||||
cfg: C,
|
cfg: C,
|
||||||
) -> ContainerFactory<'_, F, R, C> {
|
) -> ContainerFactory<'_, F, R, C> {
|
||||||
|
@ -62,9 +66,15 @@ where
|
||||||
_t: marker::PhantomData,
|
_t: marker::PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn into_service(self) -> Option<S> {
|
||||||
|
let svc = self.svc.clone();
|
||||||
|
drop(self);
|
||||||
|
Rc::try_unwrap(svc).ok()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S, R> Clone for Container<S, R> {
|
impl<S> Clone for Container<S> {
|
||||||
fn clone(&self) -> Self {
|
fn clone(&self) -> Self {
|
||||||
let index = self.waiters.borrow_mut().insert(None);
|
let index = self.waiters.borrow_mut().insert(None);
|
||||||
|
|
||||||
|
@ -72,21 +82,26 @@ impl<S, R> Clone for Container<S, R> {
|
||||||
index,
|
index,
|
||||||
svc: self.svc.clone(),
|
svc: self.svc.clone(),
|
||||||
waiters: self.waiters.clone(),
|
waiters: self.waiters.clone(),
|
||||||
_t: marker::PhantomData,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S, R> From<S> for Container<S, R>
|
impl<S> From<S> for Container<S> {
|
||||||
where
|
|
||||||
S: Service<R>,
|
|
||||||
{
|
|
||||||
fn from(svc: S) -> Self {
|
fn from(svc: S) -> Self {
|
||||||
Container::new(svc)
|
Container::new(svc)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S, R> Drop for Container<S, R> {
|
impl<S> ops::Deref for Container<S> {
|
||||||
|
type Target = S;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn deref(&self) -> &S {
|
||||||
|
self.svc.as_ref()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S> Drop for Container<S> {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
let mut waiters = self.waiters.borrow_mut();
|
let mut waiters = self.waiters.borrow_mut();
|
||||||
|
|
||||||
|
@ -268,7 +283,7 @@ impl<'f, F, R, C> Future for ContainerFactory<'f, F, R, C>
|
||||||
where
|
where
|
||||||
F: ServiceFactory<R, C> + 'f,
|
F: ServiceFactory<R, C> + 'f,
|
||||||
{
|
{
|
||||||
type Output = Result<Container<F::Service, R>, F::InitError>;
|
type Output = Result<Container<F::Service>, F::InitError>;
|
||||||
|
|
||||||
fn poll(self: Pin<&mut Self>, cx: &mut task::Context<'_>) -> Poll<Self::Output> {
|
fn poll(self: Pin<&mut Self>, cx: &mut task::Context<'_>) -> Poll<Self::Output> {
|
||||||
Poll::Ready(Ok(Container::new(task::ready!(self
|
Poll::Ready(Ok(Container::new(task::ready!(self
|
||||||
|
@ -297,10 +312,7 @@ mod tests {
|
||||||
self.1.poll_ready(cx).map(|_| Ok(()))
|
self.1.poll_ready(cx).map(|_| Ok(()))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call<'a>(&'a self, req: &'static str, _: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: &'static str, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
&'static str: 'a,
|
|
||||||
{
|
|
||||||
Ready::Ok(req)
|
Ready::Ok(req)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -128,10 +128,7 @@ where
|
||||||
type Future<'f> = Fut where Self: 'f, Req: 'f;
|
type Future<'f> = Fut where Self: 'f, Req: 'f;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'a>(&'a self, req: Req, _: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: Req, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
Req: 'a,
|
|
||||||
{
|
|
||||||
(self.f)(req)
|
(self.f)(req)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -193,10 +190,7 @@ where
|
||||||
type Future<'f> = Fut where Self: 'f;
|
type Future<'f> = Fut where Self: 'f;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'a>(&'a self, req: Req, _: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: Req, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
Req: 'a,
|
|
||||||
{
|
|
||||||
(self.f)(req)
|
(self.f)(req)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -60,10 +60,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'a>(&'a self, req: Req, _: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: Req, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
Req: 'a,
|
|
||||||
{
|
|
||||||
ready(Ok(req))
|
ready(Ok(req))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -54,7 +54,7 @@ pub use self::pipeline::{pipeline, pipeline_factory, Pipeline, PipelineFactory};
|
||||||
/// simple API surfaces. This leads to simpler design of each service, improves test-ability and
|
/// simple API surfaces. This leads to simpler design of each service, improves test-ability and
|
||||||
/// makes composition easier.
|
/// makes composition easier.
|
||||||
///
|
///
|
||||||
/// ```rust,ignore
|
/// ```rust
|
||||||
/// # use std::convert::Infallible;
|
/// # use std::convert::Infallible;
|
||||||
/// # use std::future::Future;
|
/// # use std::future::Future;
|
||||||
/// # use std::pin::Pin;
|
/// # use std::pin::Pin;
|
||||||
|
@ -101,9 +101,7 @@ pub trait Service<Req> {
|
||||||
///
|
///
|
||||||
/// Invoking `call` without first invoking `poll_ready` is permitted. Implementations must be
|
/// Invoking `call` without first invoking `poll_ready` is permitted. Implementations must be
|
||||||
/// resilient to this fact.
|
/// resilient to this fact.
|
||||||
fn call<'a>(&'a self, req: Req, ctx: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: Req, ctx: Ctx<'a, Self>) -> Self::Future<'a>;
|
||||||
where
|
|
||||||
Req: 'a;
|
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
/// Returns `Ready` when the service is able to process requests.
|
/// Returns `Ready` when the service is able to process requests.
|
||||||
|
@ -203,7 +201,7 @@ pub trait ServiceFactory<Req, Cfg = ()> {
|
||||||
where
|
where
|
||||||
Self: Sized,
|
Self: Sized,
|
||||||
{
|
{
|
||||||
Container::<Self::Service, Req>::create(self, cfg)
|
Container::<Self::Service>::create(self, cfg)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
@ -260,10 +258,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'s>(&'s self, request: Req, ctx: Ctx<'s, Self>) -> S::Future<'s>
|
fn call<'s>(&'s self, request: Req, ctx: Ctx<'s, Self>) -> S::Future<'s> {
|
||||||
where
|
|
||||||
Req: 's,
|
|
||||||
{
|
|
||||||
ctx.call_nowait(&**self, request)
|
ctx.call_nowait(&**self, request)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -287,10 +282,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'a>(&'a self, request: Req, ctx: Ctx<'a, Self>) -> S::Future<'a>
|
fn call<'a>(&'a self, request: Req, ctx: Ctx<'a, Self>) -> S::Future<'a> {
|
||||||
where
|
|
||||||
Req: 'a,
|
|
||||||
{
|
|
||||||
ctx.call_nowait(&**self, request)
|
ctx.call_nowait(&**self, request)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -54,10 +54,7 @@ where
|
||||||
crate::forward_poll_shutdown!(service);
|
crate::forward_poll_shutdown!(service);
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'a>(&'a self, req: Req, ctx: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: Req, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
Req: 'a,
|
|
||||||
{
|
|
||||||
MapFuture {
|
MapFuture {
|
||||||
fut: ctx.call(&self.service, req),
|
fut: ctx.call(&self.service, req),
|
||||||
slf: self,
|
slf: self,
|
||||||
|
@ -209,10 +206,7 @@ mod tests {
|
||||||
Poll::Ready(Ok(()))
|
Poll::Ready(Ok(()))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call<'a>(&'a self, _: (), _: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, _: (), _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
(): 'a,
|
|
||||||
{
|
|
||||||
Ready::Ok(())
|
Ready::Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -57,10 +57,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'a>(&'a self, req: R, ctx: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: R, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
R: 'a,
|
|
||||||
{
|
|
||||||
MapErrFuture {
|
MapErrFuture {
|
||||||
slf: self,
|
slf: self,
|
||||||
fut: ctx.call(&self.service, req),
|
fut: ctx.call(&self.service, req),
|
||||||
|
@ -217,10 +214,7 @@ mod tests {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call<'a>(&'a self, _: (), _: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, _: (), _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
(): 'a,
|
|
||||||
{
|
|
||||||
Ready::Err(())
|
Ready::Err(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -239,10 +239,7 @@ mod tests {
|
||||||
self.0.poll_ready(cx)
|
self.0.poll_ready(cx)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call<'a>(&'a self, req: R, ctx: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: R, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
R: 'a,
|
|
||||||
{
|
|
||||||
ctx.call(&self.0, req)
|
ctx.call(&self.0, req)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -118,7 +118,7 @@ impl<Req, Svc: Service<Req>> Pipeline<Req, Svc> {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Create service container
|
/// Create service container
|
||||||
pub fn container(self) -> Container<Svc, Req> {
|
pub fn container(self) -> Container<Svc> {
|
||||||
Container::new(self.service)
|
Container::new(self.service)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -144,10 +144,7 @@ impl<Req, Svc: Service<Req>> Service<Req> for Pipeline<Req, Svc> {
|
||||||
crate::forward_poll_shutdown!(service);
|
crate::forward_poll_shutdown!(service);
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'a>(&'a self, req: Req, ctx: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: Req, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
Req: 'a,
|
|
||||||
{
|
|
||||||
ctx.call(&self.service, req)
|
ctx.call(&self.service, req)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -60,10 +60,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call<'a>(&'a self, req: R, ctx: Ctx<'a, Self>) -> Self::Future<'a>
|
fn call<'a>(&'a self, req: R, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
R: 'a,
|
|
||||||
{
|
|
||||||
ThenServiceResponse {
|
ThenServiceResponse {
|
||||||
slf: self,
|
slf: self,
|
||||||
state: State::A {
|
state: State::A {
|
||||||
|
@ -270,10 +267,7 @@ mod tests {
|
||||||
&'a self,
|
&'a self,
|
||||||
req: Result<&'static str, &'static str>,
|
req: Result<&'static str, &'static str>,
|
||||||
_: Ctx<'a, Self>,
|
_: Ctx<'a, Self>,
|
||||||
) -> Self::Future<'a>
|
) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
Result<&'static str, &'static str>: 'a,
|
|
||||||
{
|
|
||||||
match req {
|
match req {
|
||||||
Ok(msg) => Ready::Ok(msg),
|
Ok(msg) => Ready::Ok(msg),
|
||||||
Err(_) => Ready::Err(()),
|
Err(_) => Ready::Err(()),
|
||||||
|
@ -298,10 +292,7 @@ mod tests {
|
||||||
&'a self,
|
&'a self,
|
||||||
req: Result<&'static str, ()>,
|
req: Result<&'static str, ()>,
|
||||||
_: Ctx<'a, Self>,
|
_: Ctx<'a, Self>,
|
||||||
) -> Self::Future<'a>
|
) -> Self::Future<'a> {
|
||||||
where
|
|
||||||
Result<&'static str, ()>: 'a,
|
|
||||||
{
|
|
||||||
match req {
|
match req {
|
||||||
Ok(msg) => Ready::Ok((msg, "ok")),
|
Ok(msg) => Ready::Ok((msg, "ok")),
|
||||||
Err(()) => Ready::Ok(("srv2", "err")),
|
Err(()) => Ready::Ok(("srv2", "err")),
|
||||||
|
@ -329,7 +320,6 @@ mod tests {
|
||||||
.container();
|
.container();
|
||||||
|
|
||||||
let res = srv.call(Ok("srv1")).await;
|
let res = srv.call(Ok("srv1")).await;
|
||||||
println!("=========== {:?}", res);
|
|
||||||
assert!(res.is_ok());
|
assert!(res.is_ok());
|
||||||
assert_eq!(res.unwrap(), ("srv1", "ok"));
|
assert_eq!(res.unwrap(), ("srv1", "ok"));
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-tls"
|
name = "ntex-tls"
|
||||||
version = "0.2.4"
|
version = "0.3.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "An implementation of SSL streams for ntex backed by OpenSSL"
|
description = "An implementation of SSL streams for ntex backed by OpenSSL"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -26,20 +26,20 @@ rustls = ["tls_rust"]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-bytes = "0.1.19"
|
ntex-bytes = "0.1.19"
|
||||||
ntex-io = "0.2.7"
|
ntex-io = "0.3.0"
|
||||||
ntex-util = "0.2.0"
|
ntex-util = "0.3.0"
|
||||||
ntex-service = "1.0.0"
|
ntex-service = "1.2.0"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
pin-project-lite = "0.2"
|
pin-project-lite = "0.2"
|
||||||
|
|
||||||
# openssl
|
# openssl
|
||||||
tls_openssl = { version="0.10", package = "openssl", optional = true }
|
tls_openssl = { version = "0.10", package = "openssl", optional = true }
|
||||||
|
|
||||||
# rustls
|
# rustls
|
||||||
tls_rust = { version = "0.21", package = "rustls", optional = true }
|
tls_rust = { version = "0.21", package = "rustls", optional = true }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
ntex = { version = "0.6.3", features = ["openssl", "rustls", "tokio"] }
|
ntex = { version = "0.7.0", features = ["openssl", "rustls", "tokio"] }
|
||||||
env_logger = "0.10"
|
env_logger = "0.10"
|
||||||
rustls-pemfile = { version = "1.0" }
|
rustls-pemfile = { version = "1.0" }
|
||||||
webpki-roots = { version = "0.22" }
|
webpki-roots = { version = "0.23" }
|
||||||
|
|
|
@ -2,7 +2,7 @@ use std::task::{Context, Poll};
|
||||||
use std::{error::Error, future::Future, marker::PhantomData, pin::Pin};
|
use std::{error::Error, future::Future, marker::PhantomData, pin::Pin};
|
||||||
|
|
||||||
use ntex_io::{Filter, FilterFactory, Io, Layer};
|
use ntex_io::{Filter, FilterFactory, Io, Layer};
|
||||||
use ntex_service::{Service, ServiceFactory};
|
use ntex_service::{Ctx, Service, ServiceFactory};
|
||||||
use ntex_util::{future::Ready, time::Millis};
|
use ntex_util::{future::Ready, time::Millis};
|
||||||
use tls_openssl::ssl::SslAcceptor;
|
use tls_openssl::ssl::SslAcceptor;
|
||||||
|
|
||||||
|
@ -95,7 +95,7 @@ impl<F: Filter> Service<Io<F>> for AcceptorService<F> {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: Io<F>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: Io<F>, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
AcceptorServiceResponse {
|
AcceptorServiceResponse {
|
||||||
_guard: self.conns.get(),
|
_guard: self.conns.get(),
|
||||||
fut: self.acceptor.clone().create(req),
|
fut: self.acceptor.clone().create(req),
|
||||||
|
|
|
@ -4,7 +4,7 @@ use std::{future::Future, io, marker::PhantomData, pin::Pin, sync::Arc};
|
||||||
use tls_rust::ServerConfig;
|
use tls_rust::ServerConfig;
|
||||||
|
|
||||||
use ntex_io::{Filter, FilterFactory, Io, Layer};
|
use ntex_io::{Filter, FilterFactory, Io, Layer};
|
||||||
use ntex_service::{Service, ServiceFactory};
|
use ntex_service::{Ctx, Service, ServiceFactory};
|
||||||
use ntex_util::{future::Ready, time::Millis};
|
use ntex_util::{future::Ready, time::Millis};
|
||||||
|
|
||||||
use super::{TlsAcceptor, TlsFilter};
|
use super::{TlsAcceptor, TlsFilter};
|
||||||
|
@ -93,7 +93,7 @@ impl<F: Filter> Service<Io<F>> for AcceptorService<F> {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: Io<F>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: Io<F>, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
AcceptorServiceFut {
|
AcceptorServiceFut {
|
||||||
_guard: self.conns.get(),
|
_guard: self.conns.get(),
|
||||||
fut: self.acceptor.clone().create(req),
|
fut: self.acceptor.clone().create(req),
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-tokio"
|
name = "ntex-tokio"
|
||||||
version = "0.2.3"
|
version = "0.3.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "tokio intergration for ntex framework"
|
description = "tokio intergration for ntex framework"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -17,8 +17,8 @@ path = "src/lib.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-bytes = "0.1.19"
|
ntex-bytes = "0.1.19"
|
||||||
ntex-io = "0.2.4"
|
ntex-io = "0.3.0"
|
||||||
ntex-util = "0.2.0"
|
ntex-util = "0.3.0"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
pin-project-lite = "0.2"
|
pin-project-lite = "0.2"
|
||||||
tokio = { version = "1", default-features = false, features = ["rt", "net", "sync", "signal"] }
|
tokio = { version = "1", default-features = false, features = ["rt", "net", "sync", "signal"] }
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
|
## [0.3.0] - 2023-06-xx
|
||||||
|
|
||||||
|
* Upgrade to ntex-service 1.2
|
||||||
|
|
||||||
## [0.2.3] - 2023-06-04
|
## [0.2.3] - 2023-06-04
|
||||||
|
|
||||||
* Refactor timer driver
|
* Refactor timer driver
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-util"
|
name = "ntex-util"
|
||||||
version = "0.2.3"
|
version = "0.3.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "Utilities for ntex framework"
|
description = "Utilities for ntex framework"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -17,7 +17,7 @@ path = "src/lib.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-rt = "0.4.7"
|
ntex-rt = "0.4.7"
|
||||||
ntex-service = "1.0.0"
|
ntex-service = "1.2.0"
|
||||||
bitflags = "1.3"
|
bitflags = "1.3"
|
||||||
fxhash = "0.2.1"
|
fxhash = "0.2.1"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
|
@ -28,7 +28,7 @@ futures-sink = { version = "0.3", default-features = false, features = ["alloc"]
|
||||||
pin-project-lite = "0.2.9"
|
pin-project-lite = "0.2.9"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
ntex = { version = "0.6.0", features = ["tokio"] }
|
ntex = { version = "0.7.0", features = ["tokio"] }
|
||||||
ntex-bytes = "0.1.18"
|
ntex-bytes = "0.1.18"
|
||||||
ntex-macros = "0.1.3"
|
ntex-macros = "0.1.3"
|
||||||
futures-util = { version = "0.3", default-features = false, features = ["alloc"] }
|
futures-util = { version = "0.3", default-features = false, features = ["alloc"] }
|
||||||
|
|
|
@ -1,63 +1,59 @@
|
||||||
//! Service that buffers incomming requests.
|
//! Service that buffers incomming requests.
|
||||||
use std::cell::{Cell, RefCell};
|
use std::cell::{Cell, RefCell};
|
||||||
use std::task::{Context, Poll};
|
use std::task::{ready, Context, Poll};
|
||||||
use std::{collections::VecDeque, future::Future, marker::PhantomData, pin::Pin, rc::Rc};
|
use std::{collections::VecDeque, future::Future, marker::PhantomData, pin::Pin};
|
||||||
|
|
||||||
use ntex_service::{IntoService, Middleware, Service};
|
use ntex_service::{Ctx, IntoService, Middleware, Service, ServiceCall};
|
||||||
|
|
||||||
use crate::{channel::oneshot, future::Either, task::LocalWaker};
|
use crate::{channel::oneshot, future::Either, task::LocalWaker};
|
||||||
|
|
||||||
/// Buffer - service factory for service that can buffer incoming request.
|
/// Buffer - service factory for service that can buffer incoming request.
|
||||||
///
|
///
|
||||||
/// Default number of buffered requests is 16
|
/// Default number of buffered requests is 16
|
||||||
pub struct Buffer<R, E> {
|
pub struct Buffer<R> {
|
||||||
buf_size: usize,
|
buf_size: usize,
|
||||||
err: Rc<dyn Fn() -> E>,
|
|
||||||
_t: PhantomData<R>,
|
_t: PhantomData<R>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<R, E> Buffer<R, E> {
|
impl<R> Default for Buffer<R> {
|
||||||
pub fn new<F>(f: F) -> Self
|
fn default() -> Self {
|
||||||
where
|
|
||||||
F: Fn() -> E + 'static,
|
|
||||||
{
|
|
||||||
Self {
|
Self {
|
||||||
buf_size: 16,
|
buf_size: 16,
|
||||||
err: Rc::new(f),
|
|
||||||
_t: PhantomData,
|
_t: PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<R> Buffer<R> {
|
||||||
pub fn buf_size(mut self, size: usize) -> Self {
|
pub fn buf_size(mut self, size: usize) -> Self {
|
||||||
self.buf_size = size;
|
self.buf_size = size;
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<R, E> Clone for Buffer<R, E> {
|
impl<R> Clone for Buffer<R> {
|
||||||
fn clone(&self) -> Self {
|
fn clone(&self) -> Self {
|
||||||
Self {
|
Self {
|
||||||
buf_size: self.buf_size,
|
buf_size: self.buf_size,
|
||||||
err: self.err.clone(),
|
|
||||||
_t: PhantomData,
|
_t: PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<R, S, E> Middleware<S> for Buffer<R, E>
|
impl<R, S> Middleware<S> for Buffer<R>
|
||||||
where
|
where
|
||||||
S: Service<R, Error = E>,
|
S: Service<R>,
|
||||||
{
|
{
|
||||||
type Service = BufferService<R, S, E>;
|
type Service = BufferService<R, S>;
|
||||||
|
|
||||||
fn create(&self, service: S) -> Self::Service {
|
fn create(&self, service: S) -> Self::Service {
|
||||||
BufferService {
|
BufferService {
|
||||||
service,
|
service,
|
||||||
size: self.buf_size,
|
size: self.buf_size,
|
||||||
err: self.err.clone(),
|
|
||||||
ready: Cell::new(false),
|
ready: Cell::new(false),
|
||||||
waker: LocalWaker::default(),
|
waker: LocalWaker::default(),
|
||||||
buf: RefCell::new(VecDeque::with_capacity(self.buf_size)),
|
buf: RefCell::new(VecDeque::with_capacity(self.buf_size)),
|
||||||
|
_t: PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -65,58 +61,57 @@ where
|
||||||
/// Buffer service - service that can buffer incoming requests.
|
/// Buffer service - service that can buffer incoming requests.
|
||||||
///
|
///
|
||||||
/// Default number of buffered requests is 16
|
/// Default number of buffered requests is 16
|
||||||
pub struct BufferService<R, S: Service<R, Error = E>, E> {
|
pub struct BufferService<R, S: Service<R>> {
|
||||||
size: usize,
|
size: usize,
|
||||||
ready: Cell<bool>,
|
ready: Cell<bool>,
|
||||||
service: S,
|
service: S,
|
||||||
waker: LocalWaker,
|
waker: LocalWaker,
|
||||||
err: Rc<dyn Fn() -> E>,
|
buf: RefCell<VecDeque<oneshot::Sender<()>>>,
|
||||||
buf: RefCell<VecDeque<(oneshot::Sender<R>, R)>>,
|
_t: PhantomData<R>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<R, S, E> BufferService<R, S, E>
|
impl<R, S> BufferService<R, S>
|
||||||
where
|
where
|
||||||
S: Service<R, Error = E>,
|
S: Service<R>,
|
||||||
{
|
{
|
||||||
pub fn new<U, F>(size: usize, err: F, service: U) -> Self
|
pub fn new<U>(size: usize, service: U) -> Self
|
||||||
where
|
where
|
||||||
U: IntoService<S, R>,
|
U: IntoService<S, R>,
|
||||||
F: Fn() -> E + 'static,
|
|
||||||
{
|
{
|
||||||
Self {
|
Self {
|
||||||
size,
|
size,
|
||||||
err: Rc::new(err),
|
|
||||||
ready: Cell::new(false),
|
ready: Cell::new(false),
|
||||||
service: service.into_service(),
|
service: service.into_service(),
|
||||||
waker: LocalWaker::default(),
|
waker: LocalWaker::default(),
|
||||||
buf: RefCell::new(VecDeque::with_capacity(size)),
|
buf: RefCell::new(VecDeque::with_capacity(size)),
|
||||||
|
_t: PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<R, S, E> Clone for BufferService<R, S, E>
|
impl<R, S> Clone for BufferService<R, S>
|
||||||
where
|
where
|
||||||
S: Service<R, Error = E> + Clone,
|
S: Service<R> + Clone,
|
||||||
{
|
{
|
||||||
fn clone(&self) -> Self {
|
fn clone(&self) -> Self {
|
||||||
Self {
|
Self {
|
||||||
size: self.size,
|
size: self.size,
|
||||||
err: self.err.clone(),
|
|
||||||
ready: Cell::new(false),
|
ready: Cell::new(false),
|
||||||
service: self.service.clone(),
|
service: self.service.clone(),
|
||||||
waker: LocalWaker::default(),
|
waker: LocalWaker::default(),
|
||||||
buf: RefCell::new(VecDeque::with_capacity(self.size)),
|
buf: RefCell::new(VecDeque::with_capacity(self.size)),
|
||||||
|
_t: PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<R, S, E> Service<R> for BufferService<R, S, E>
|
impl<R, S> Service<R> for BufferService<R, S>
|
||||||
where
|
where
|
||||||
S: Service<R, Error = E>,
|
S: Service<R>,
|
||||||
{
|
{
|
||||||
type Response = S::Response;
|
type Response = S::Response;
|
||||||
type Error = S::Error;
|
type Error = S::Error;
|
||||||
type Future<'f> = Either<S::Future<'f>, BufferServiceResponse<'f, R, S, E>> where Self: 'f, R: 'f;
|
type Future<'f> = Either<ServiceCall<'f, S, R>, BufferServiceResponse<'f, R, S>> where Self: 'f, R: 'f;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
||||||
|
@ -132,8 +127,8 @@ where
|
||||||
log::trace!("Buffer limit exceeded");
|
log::trace!("Buffer limit exceeded");
|
||||||
Poll::Pending
|
Poll::Pending
|
||||||
}
|
}
|
||||||
} else if let Some((sender, req)) = buffer.pop_front() {
|
} else if let Some(sender) = buffer.pop_front() {
|
||||||
let _ = sender.send(req);
|
let _ = sender.send(());
|
||||||
self.ready.set(false);
|
self.ready.set(false);
|
||||||
Poll::Ready(Ok(()))
|
Poll::Ready(Ok(()))
|
||||||
} else {
|
} else {
|
||||||
|
@ -143,17 +138,18 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: R) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: R, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
if self.ready.get() {
|
if self.ready.get() {
|
||||||
self.ready.set(false);
|
self.ready.set(false);
|
||||||
Either::Left(self.service.call(req))
|
Either::Left(ctx.call(&self.service, req))
|
||||||
} else {
|
} else {
|
||||||
let (tx, rx) = oneshot::channel();
|
let (tx, rx) = oneshot::channel();
|
||||||
self.buf.borrow_mut().push_back((tx, req));
|
self.buf.borrow_mut().push_back(tx);
|
||||||
|
|
||||||
Either::Right(BufferServiceResponse {
|
Either::Right(BufferServiceResponse {
|
||||||
slf: self,
|
slf: self,
|
||||||
state: State::Tx { rx },
|
fut: ctx.call(&self.service, req),
|
||||||
|
rx: Some(rx),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -163,63 +159,40 @@ where
|
||||||
|
|
||||||
pin_project_lite::pin_project! {
|
pin_project_lite::pin_project! {
|
||||||
#[doc(hidden)]
|
#[doc(hidden)]
|
||||||
pub struct BufferServiceResponse<'f, R, S: Service<R, Error = E>, E>
|
#[must_use = "futures do nothing unless polled"]
|
||||||
|
pub struct BufferServiceResponse<'f, R, S: Service<R>>
|
||||||
{
|
{
|
||||||
slf: &'f BufferService<R, S, E>,
|
|
||||||
#[pin]
|
#[pin]
|
||||||
state: State<R, S::Future<'f>>,
|
fut: ServiceCall<'f, S, R>,
|
||||||
|
slf: &'f BufferService<R, S>,
|
||||||
|
rx: Option<oneshot::Receiver<()>>,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pin_project_lite::pin_project! {
|
impl<'f, R, S> Future for BufferServiceResponse<'f, R, S>
|
||||||
#[project = StateProject]
|
|
||||||
enum State<R, F>
|
|
||||||
where F: Future,
|
|
||||||
{
|
|
||||||
Tx { rx: oneshot::Receiver<R> },
|
|
||||||
Srv { #[pin] fut: F },
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'f, R, S, E> Future for BufferServiceResponse<'f, R, S, E>
|
|
||||||
where
|
where
|
||||||
S: Service<R, Error = E>,
|
S: Service<R>,
|
||||||
{
|
{
|
||||||
type Output = Result<S::Response, S::Error>;
|
type Output = Result<S::Response, S::Error>;
|
||||||
|
|
||||||
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
||||||
let mut this = self.as_mut().project();
|
let this = self.as_mut().project();
|
||||||
|
|
||||||
loop {
|
if let Some(ref rx) = this.rx {
|
||||||
match this.state.project() {
|
let _ = ready!(rx.poll_recv(cx));
|
||||||
StateProject::Tx { rx } => match Pin::new(rx).poll(cx) {
|
this.rx.take();
|
||||||
Poll::Ready(Ok(req)) => {
|
|
||||||
let state = State::Srv {
|
|
||||||
fut: this.slf.service.call(req),
|
|
||||||
};
|
|
||||||
this = self.as_mut().project();
|
|
||||||
this.state.set(state);
|
|
||||||
}
|
|
||||||
Poll::Ready(Err(_)) => return Poll::Ready(Err((*this.slf.err)())),
|
|
||||||
Poll::Pending => return Poll::Pending,
|
|
||||||
},
|
|
||||||
StateProject::Srv { fut } => {
|
|
||||||
let res = match fut.poll(cx) {
|
|
||||||
Poll::Ready(res) => res,
|
|
||||||
Poll::Pending => return Poll::Pending,
|
|
||||||
};
|
|
||||||
this.slf.waker.wake();
|
|
||||||
return Poll::Ready(res);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let res = ready!(this.fut.poll(cx));
|
||||||
|
this.slf.waker.wake();
|
||||||
|
Poll::Ready(res)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use ntex_service::{apply, fn_factory, Service, ServiceFactory};
|
use ntex_service::{apply, fn_factory, Container, Service, ServiceFactory};
|
||||||
use std::task::{Context, Poll};
|
use std::{rc::Rc, task::Context, task::Poll, time::Duration};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::future::{lazy, Ready};
|
use crate::future::{lazy, Ready};
|
||||||
|
@ -247,7 +220,7 @@ mod tests {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, _: ()) -> Self::Future<'_> {
|
fn call<'a>(&'a self, _: (), _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
self.0.ready.set(false);
|
self.0.ready.set(false);
|
||||||
self.0.count.set(self.0.count.get() + 1);
|
self.0.count.set(self.0.count.get() + 1);
|
||||||
Ready::Ok(())
|
Ready::Ok(())
|
||||||
|
@ -255,21 +228,29 @@ mod tests {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[ntex_macros::rt_test2]
|
#[ntex_macros::rt_test2]
|
||||||
async fn test_transform() {
|
async fn test_service() {
|
||||||
let inner = Rc::new(Inner {
|
let inner = Rc::new(Inner {
|
||||||
ready: Cell::new(false),
|
ready: Cell::new(false),
|
||||||
waker: LocalWaker::default(),
|
waker: LocalWaker::default(),
|
||||||
count: Cell::new(0),
|
count: Cell::new(0),
|
||||||
});
|
});
|
||||||
|
|
||||||
let srv = BufferService::new(2, || (), TestService(inner.clone())).clone();
|
let srv = Container::new(BufferService::new(2, TestService(inner.clone())).clone());
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let fut1 = srv.call(());
|
let srv1 = srv.clone();
|
||||||
|
ntex::rt::spawn(async move {
|
||||||
|
let _ = srv1.call(()).await;
|
||||||
|
});
|
||||||
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(inner.count.get(), 0);
|
assert_eq!(inner.count.get(), 0);
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let fut2 = srv.call(());
|
let srv1 = srv.clone();
|
||||||
|
ntex::rt::spawn(async move {
|
||||||
|
let _ = srv1.call(()).await;
|
||||||
|
});
|
||||||
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(inner.count.get(), 0);
|
assert_eq!(inner.count.get(), 0);
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
||||||
|
|
||||||
|
@ -277,14 +258,14 @@ mod tests {
|
||||||
inner.waker.wake();
|
inner.waker.wake();
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let _ = fut1.await;
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(inner.count.get(), 1);
|
assert_eq!(inner.count.get(), 1);
|
||||||
|
|
||||||
inner.ready.set(true);
|
inner.ready.set(true);
|
||||||
inner.waker.wake();
|
inner.waker.wake();
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let _ = fut2.await;
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(inner.count.get(), 2);
|
assert_eq!(inner.count.get(), 2);
|
||||||
|
|
||||||
let inner = Rc::new(Inner {
|
let inner = Rc::new(Inner {
|
||||||
|
@ -293,7 +274,7 @@ mod tests {
|
||||||
count: Cell::new(0),
|
count: Cell::new(0),
|
||||||
});
|
});
|
||||||
|
|
||||||
let srv = BufferService::new(2, || (), TestService(inner.clone()));
|
let srv = Container::new(BufferService::new(2, TestService(inner.clone())));
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
let _ = srv.call(()).await;
|
let _ = srv.call(()).await;
|
||||||
assert_eq!(inner.count.get(), 1);
|
assert_eq!(inner.count.get(), 1);
|
||||||
|
@ -303,7 +284,7 @@ mod tests {
|
||||||
|
|
||||||
#[ntex_macros::rt_test2]
|
#[ntex_macros::rt_test2]
|
||||||
#[allow(clippy::redundant_clone)]
|
#[allow(clippy::redundant_clone)]
|
||||||
async fn test_newtransform() {
|
async fn test_middleware() {
|
||||||
let inner = Rc::new(Inner {
|
let inner = Rc::new(Inner {
|
||||||
ready: Cell::new(false),
|
ready: Cell::new(false),
|
||||||
waker: LocalWaker::default(),
|
waker: LocalWaker::default(),
|
||||||
|
@ -311,18 +292,26 @@ mod tests {
|
||||||
});
|
});
|
||||||
|
|
||||||
let srv = apply(
|
let srv = apply(
|
||||||
Buffer::new(|| ()).buf_size(2).clone(),
|
Buffer::default().buf_size(2).clone(),
|
||||||
fn_factory(|| async { Ok::<_, ()>(TestService(inner.clone())) }),
|
fn_factory(|| async { Ok::<_, ()>(TestService(inner.clone())) }),
|
||||||
);
|
);
|
||||||
|
|
||||||
let srv = srv.create(&()).await.unwrap();
|
let srv = srv.container(&()).await.unwrap();
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let fut1 = srv.call(());
|
let srv1 = srv.clone();
|
||||||
|
ntex::rt::spawn(async move {
|
||||||
|
let _ = srv1.call(()).await;
|
||||||
|
});
|
||||||
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(inner.count.get(), 0);
|
assert_eq!(inner.count.get(), 0);
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let fut2 = srv.call(());
|
let srv1 = srv.clone();
|
||||||
|
ntex::rt::spawn(async move {
|
||||||
|
let _ = srv1.call(()).await;
|
||||||
|
});
|
||||||
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(inner.count.get(), 0);
|
assert_eq!(inner.count.get(), 0);
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
||||||
|
|
||||||
|
@ -330,14 +319,14 @@ mod tests {
|
||||||
inner.waker.wake();
|
inner.waker.wake();
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let _ = fut1.await;
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(inner.count.get(), 1);
|
assert_eq!(inner.count.get(), 1);
|
||||||
|
|
||||||
inner.ready.set(true);
|
inner.ready.set(true);
|
||||||
inner.waker.wake();
|
inner.waker.wake();
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let _ = fut2.await;
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(inner.count.get(), 2);
|
assert_eq!(inner.count.get(), 2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
//! Service that limits number of in-flight async requests.
|
//! Service that limits number of in-flight async requests.
|
||||||
use std::{future::Future, marker::PhantomData, pin::Pin, task::Context, task::Poll};
|
use std::{future::Future, marker::PhantomData, pin::Pin, task::Context, task::Poll};
|
||||||
|
|
||||||
use ntex_service::{IntoService, Middleware, Service};
|
use ntex_service::{Ctx, IntoService, Middleware, Service, ServiceCall};
|
||||||
|
|
||||||
use super::counter::{Counter, CounterGuard};
|
use super::counter::{Counter, CounterGuard};
|
||||||
|
|
||||||
|
@ -76,9 +76,9 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: R) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: R, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
InFlightServiceResponse {
|
InFlightServiceResponse {
|
||||||
fut: self.service.call(req),
|
fut: ctx.call(&self.service, req),
|
||||||
_guard: self.count.get(),
|
_guard: self.count.get(),
|
||||||
_t: PhantomData,
|
_t: PhantomData,
|
||||||
}
|
}
|
||||||
|
@ -93,7 +93,7 @@ pin_project_lite::pin_project! {
|
||||||
where T: 'f, R: 'f
|
where T: 'f, R: 'f
|
||||||
{
|
{
|
||||||
#[pin]
|
#[pin]
|
||||||
fut: T::Future<'f>,
|
fut: ServiceCall<'f, T, R>,
|
||||||
_guard: CounterGuard,
|
_guard: CounterGuard,
|
||||||
_t: PhantomData<R>
|
_t: PhantomData<R>
|
||||||
}
|
}
|
||||||
|
@ -109,39 +109,43 @@ impl<'f, T: Service<R>, R> Future for InFlightServiceResponse<'f, T, R> {
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use ntex_service::{apply, fn_factory, Service, ServiceFactory};
|
use ntex_service::{apply, fn_factory, Container, Ctx, Service, ServiceFactory};
|
||||||
use std::{task::Poll, time::Duration};
|
use std::{cell::RefCell, task::Poll, time::Duration};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::future::{lazy, BoxFuture};
|
use crate::{channel::oneshot, future::lazy, future::BoxFuture};
|
||||||
|
|
||||||
struct SleepService(Duration);
|
struct SleepService(oneshot::Receiver<()>);
|
||||||
|
|
||||||
impl Service<()> for SleepService {
|
impl Service<()> for SleepService {
|
||||||
type Response = ();
|
type Response = ();
|
||||||
type Error = ();
|
type Error = ();
|
||||||
type Future<'f> = BoxFuture<'f, Result<(), ()>>;
|
type Future<'f> = BoxFuture<'f, Result<(), ()>>;
|
||||||
|
|
||||||
fn call(&self, _: ()) -> Self::Future<'_> {
|
fn call<'a>(&'a self, _: (), _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
let fut = crate::time::sleep(self.0);
|
|
||||||
Box::pin(async move {
|
Box::pin(async move {
|
||||||
fut.await;
|
let _ = self.0.recv().await;
|
||||||
Ok::<_, ()>(())
|
Ok::<_, ()>(())
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[ntex_macros::rt_test2]
|
#[ntex_macros::rt_test2]
|
||||||
async fn test_inflight() {
|
async fn test_service() {
|
||||||
let wait_time = Duration::from_millis(50);
|
let (tx, rx) = oneshot::channel();
|
||||||
|
|
||||||
let srv = InFlightService::new(1, SleepService(wait_time));
|
let srv = Container::new(InFlightService::new(1, SleepService(rx)));
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let res = srv.call(());
|
let srv2 = srv.clone();
|
||||||
|
ntex::rt::spawn(async move {
|
||||||
|
let _ = srv2.call(()).await;
|
||||||
|
});
|
||||||
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
||||||
|
|
||||||
let _ = res.await;
|
let _ = tx.send(());
|
||||||
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
assert!(lazy(|cx| srv.poll_shutdown(cx)).await.is_ready());
|
assert!(lazy(|cx| srv.poll_shutdown(cx)).await.is_ready());
|
||||||
}
|
}
|
||||||
|
@ -154,19 +158,28 @@ mod tests {
|
||||||
"InFlight { max_inflight: 1 }"
|
"InFlight { max_inflight: 1 }"
|
||||||
);
|
);
|
||||||
|
|
||||||
let wait_time = Duration::from_millis(50);
|
let (tx, rx) = oneshot::channel();
|
||||||
|
let rx = RefCell::new(Some(rx));
|
||||||
let srv = apply(
|
let srv = apply(
|
||||||
InFlight::new(1),
|
InFlight::new(1),
|
||||||
fn_factory(|| async { Ok::<_, ()>(SleepService(wait_time)) }),
|
fn_factory(move || {
|
||||||
|
let rx = rx.borrow_mut().take().unwrap();
|
||||||
|
async move { Ok::<_, ()>(SleepService(rx)) }
|
||||||
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
let srv = srv.create(&()).await.unwrap();
|
let srv = srv.container(&()).await.unwrap();
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let res = srv.call(());
|
let srv2 = srv.clone();
|
||||||
|
ntex::rt::spawn(async move {
|
||||||
|
let _ = srv2.call(()).await;
|
||||||
|
});
|
||||||
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
||||||
|
|
||||||
let _ = res.await;
|
let _ = tx.send(());
|
||||||
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use std::task::{Context, Poll};
|
use std::task::{Context, Poll};
|
||||||
use std::{cell::Cell, convert::Infallible, marker, time::Duration, time::Instant};
|
use std::{cell::Cell, convert::Infallible, marker, time::Duration, time::Instant};
|
||||||
|
|
||||||
use ntex_service::{Service, ServiceFactory};
|
use ntex_service::{Ctx, Service, ServiceFactory};
|
||||||
|
|
||||||
use crate::future::Ready;
|
use crate::future::Ready;
|
||||||
use crate::time::{now, sleep, Millis, Sleep};
|
use crate::time::{now, sleep, Millis, Sleep};
|
||||||
|
@ -113,7 +113,7 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, req: R) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: R, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
self.expire.set(now());
|
self.expire.set(now());
|
||||||
Ready::Ok(req)
|
Ready::Ok(req)
|
||||||
}
|
}
|
||||||
|
@ -121,7 +121,7 @@ where
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use ntex_service::{Service, ServiceFactory};
|
use ntex_service::ServiceFactory;
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::future::lazy;
|
use crate::future::lazy;
|
||||||
|
@ -134,7 +134,7 @@ mod tests {
|
||||||
let factory = KeepAlive::new(Millis(100), || TestErr);
|
let factory = KeepAlive::new(Millis(100), || TestErr);
|
||||||
let _ = factory.clone();
|
let _ = factory.clone();
|
||||||
|
|
||||||
let service = factory.create(&()).await.unwrap();
|
let service = factory.container(&()).await.unwrap();
|
||||||
|
|
||||||
assert_eq!(service.call(1usize).await, Ok(1usize));
|
assert_eq!(service.call(1usize).await, Ok(1usize));
|
||||||
assert!(lazy(|cx| service.poll_ready(cx)).await.is_ready());
|
assert!(lazy(|cx| service.poll_ready(cx)).await.is_ready());
|
||||||
|
|
|
@ -4,7 +4,6 @@ mod extensions;
|
||||||
pub mod inflight;
|
pub mod inflight;
|
||||||
pub mod keepalive;
|
pub mod keepalive;
|
||||||
pub mod onerequest;
|
pub mod onerequest;
|
||||||
pub mod shared;
|
|
||||||
pub mod timeout;
|
pub mod timeout;
|
||||||
pub mod variant;
|
pub mod variant;
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
//! Service that limits number of in-flight async requests to 1.
|
//! Service that limits number of in-flight async requests to 1.
|
||||||
use std::{cell::Cell, future::Future, pin::Pin, task::Context, task::Poll};
|
use std::{cell::Cell, future::Future, pin::Pin, task::Context, task::Poll};
|
||||||
|
|
||||||
use ntex_service::{IntoService, Middleware, Service};
|
use ntex_service::{Ctx, IntoService, Middleware, Service, ServiceCall};
|
||||||
|
|
||||||
use crate::task::LocalWaker;
|
use crate::task::LocalWaker;
|
||||||
|
|
||||||
|
@ -63,11 +63,11 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: R) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: R, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
self.ready.set(false);
|
self.ready.set(false);
|
||||||
|
|
||||||
OneRequestServiceResponse {
|
OneRequestServiceResponse {
|
||||||
fut: self.service.call(req),
|
fut: ctx.call(&self.service, req),
|
||||||
service: self,
|
service: self,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -81,7 +81,7 @@ pin_project_lite::pin_project! {
|
||||||
where T: 'f, R: 'f
|
where T: 'f, R: 'f
|
||||||
{
|
{
|
||||||
#[pin]
|
#[pin]
|
||||||
fut: T::Future<'f>,
|
fut: ServiceCall<'f, T, R>,
|
||||||
service: &'f OneRequestService<T>,
|
service: &'f OneRequestService<T>,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -101,23 +101,22 @@ impl<'f, T: Service<R>, R> Future for OneRequestServiceResponse<'f, T, R> {
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use ntex_service::{apply, fn_factory, Service, ServiceFactory};
|
use ntex_service::{apply, fn_factory, Container, Ctx, Service, ServiceFactory};
|
||||||
use std::{task::Poll, time::Duration};
|
use std::{cell::RefCell, task::Poll, time::Duration};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::future::{lazy, BoxFuture};
|
use crate::{channel::oneshot, future::lazy, future::BoxFuture};
|
||||||
|
|
||||||
struct SleepService(Duration);
|
struct SleepService(oneshot::Receiver<()>);
|
||||||
|
|
||||||
impl Service<()> for SleepService {
|
impl Service<()> for SleepService {
|
||||||
type Response = ();
|
type Response = ();
|
||||||
type Error = ();
|
type Error = ();
|
||||||
type Future<'f> = BoxFuture<'f, Result<(), ()>>;
|
type Future<'f> = BoxFuture<'f, Result<(), ()>>;
|
||||||
|
|
||||||
fn call(&self, _: ()) -> Self::Future<'_> {
|
fn call<'a>(&'a self, _: (), _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
let fut = crate::time::sleep(self.0);
|
|
||||||
Box::pin(async move {
|
Box::pin(async move {
|
||||||
fut.await;
|
let _ = self.0.recv().await;
|
||||||
Ok::<_, ()>(())
|
Ok::<_, ()>(())
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -125,15 +124,20 @@ mod tests {
|
||||||
|
|
||||||
#[ntex_macros::rt_test2]
|
#[ntex_macros::rt_test2]
|
||||||
async fn test_oneshot() {
|
async fn test_oneshot() {
|
||||||
let wait_time = Duration::from_millis(50);
|
let (tx, rx) = oneshot::channel();
|
||||||
|
|
||||||
let srv = OneRequestService::new(SleepService(wait_time));
|
let srv = Container::new(OneRequestService::new(SleepService(rx)));
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let res = srv.call(());
|
let srv2 = srv.clone();
|
||||||
|
ntex::rt::spawn(async move {
|
||||||
|
let _ = srv2.call(()).await;
|
||||||
|
});
|
||||||
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
||||||
|
|
||||||
let _ = res.await;
|
let _ = tx.send(());
|
||||||
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
assert!(lazy(|cx| srv.poll_shutdown(cx)).await.is_ready());
|
assert!(lazy(|cx| srv.poll_shutdown(cx)).await.is_ready());
|
||||||
}
|
}
|
||||||
|
@ -142,19 +146,28 @@ mod tests {
|
||||||
async fn test_middleware() {
|
async fn test_middleware() {
|
||||||
assert_eq!(format!("{:?}", OneRequest), "OneRequest");
|
assert_eq!(format!("{:?}", OneRequest), "OneRequest");
|
||||||
|
|
||||||
let wait_time = Duration::from_millis(50);
|
let (tx, rx) = oneshot::channel();
|
||||||
|
let rx = RefCell::new(Some(rx));
|
||||||
let srv = apply(
|
let srv = apply(
|
||||||
OneRequest,
|
OneRequest,
|
||||||
fn_factory(|| async { Ok::<_, ()>(SleepService(wait_time)) }),
|
fn_factory(move || {
|
||||||
|
let rx = rx.borrow_mut().take().unwrap();
|
||||||
|
async move { Ok::<_, ()>(SleepService(rx)) }
|
||||||
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
let srv = srv.create(&()).await.unwrap();
|
let srv = srv.container(&()).await.unwrap();
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let res = srv.call(());
|
let srv2 = srv.clone();
|
||||||
|
ntex::rt::spawn(async move {
|
||||||
|
let _ = srv2.call(()).await;
|
||||||
|
});
|
||||||
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
||||||
|
|
||||||
let _ = res.await;
|
let _ = tx.send(());
|
||||||
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,169 +0,0 @@
|
||||||
/// A service that can be checked for readiness by multiple tasks
|
|
||||||
use std::{
|
|
||||||
cell::Cell, cell::RefCell, marker::PhantomData, rc::Rc, task::Context, task::Poll,
|
|
||||||
};
|
|
||||||
|
|
||||||
use ntex_service::{Middleware, Service};
|
|
||||||
|
|
||||||
use crate::channel::{condition, oneshot};
|
|
||||||
use crate::future::{poll_fn, select, Either};
|
|
||||||
|
|
||||||
/// A middleware that construct sharable service
|
|
||||||
pub struct Shared<R>(PhantomData<R>);
|
|
||||||
|
|
||||||
impl<R> Shared<R> {
|
|
||||||
pub fn new() -> Self {
|
|
||||||
Self(PhantomData)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<R> Default for Shared<R> {
|
|
||||||
fn default() -> Self {
|
|
||||||
Self::new()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: Service<R>, R> Middleware<S> for Shared<R> {
|
|
||||||
type Service = SharedService<S, R>;
|
|
||||||
|
|
||||||
fn create(&self, service: S) -> Self::Service {
|
|
||||||
SharedService::new(service)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// A service that can be checked for readiness by multiple tasks
|
|
||||||
pub struct SharedService<S: Service<R>, R> {
|
|
||||||
inner: Rc<Inner<S, R>>,
|
|
||||||
readiness: condition::Waiter,
|
|
||||||
}
|
|
||||||
|
|
||||||
struct Inner<S: Service<R>, R> {
|
|
||||||
service: S,
|
|
||||||
ready: condition::Condition,
|
|
||||||
driver_stop: Cell<Option<oneshot::Sender<()>>>,
|
|
||||||
driver_running: Cell<bool>,
|
|
||||||
error: RefCell<Option<S::Error>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: Service<R>, R> SharedService<S, R> {
|
|
||||||
pub fn new(service: S) -> Self {
|
|
||||||
let condition = condition::Condition::default();
|
|
||||||
Self {
|
|
||||||
readiness: condition.wait(),
|
|
||||||
inner: Rc::new(Inner {
|
|
||||||
service,
|
|
||||||
ready: condition,
|
|
||||||
driver_stop: Cell::default(),
|
|
||||||
driver_running: Cell::default(),
|
|
||||||
error: RefCell::default(),
|
|
||||||
}),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: Service<R>, R> Clone for SharedService<S, R> {
|
|
||||||
fn clone(&self) -> Self {
|
|
||||||
Self {
|
|
||||||
inner: self.inner.clone(),
|
|
||||||
readiness: self.readiness.clone(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: Service<R>, R> Drop for SharedService<S, R> {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
if self.inner.driver_running.get() {
|
|
||||||
// the only live references to inner are in this SharedService instance and the driver task
|
|
||||||
if Rc::strong_count(&self.inner) == 2 {
|
|
||||||
if let Some(stop) = self.inner.driver_stop.take() {
|
|
||||||
let _ = stop.send(());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S, R> Service<R> for SharedService<S, R>
|
|
||||||
where
|
|
||||||
S: Service<R> + 'static,
|
|
||||||
S::Error: Clone,
|
|
||||||
R: 'static,
|
|
||||||
{
|
|
||||||
type Response = S::Response;
|
|
||||||
type Error = S::Error;
|
|
||||||
type Future<'f> = S::Future<'f> where Self: 'f, R: 'f;
|
|
||||||
|
|
||||||
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
|
||||||
// if there is an error, it should be returned to all tasks checking readiness
|
|
||||||
if let Some(error) = self.inner.error.borrow().as_ref() {
|
|
||||||
return Poll::Ready(Err(error.clone()));
|
|
||||||
}
|
|
||||||
|
|
||||||
// if the service is being driven to readiness we must register our waker and wait
|
|
||||||
if self.inner.driver_running.get() {
|
|
||||||
log::trace!("polled SharedService driver, driver is already running");
|
|
||||||
// register waker to be notified, regardless of any previous notification
|
|
||||||
let _ = self.readiness.poll_ready(cx);
|
|
||||||
return Poll::Pending;
|
|
||||||
}
|
|
||||||
|
|
||||||
// driver is not running, check the inner service is ready
|
|
||||||
let result = self.inner.service.poll_ready(cx);
|
|
||||||
log::trace!(
|
|
||||||
"polled SharedService, ready: {}, errored: {}",
|
|
||||||
result.is_ready(),
|
|
||||||
matches!(result, Poll::Ready(Err(_)))
|
|
||||||
);
|
|
||||||
|
|
||||||
match result {
|
|
||||||
// pass through service is ready, allow call
|
|
||||||
Poll::Ready(Ok(())) => Poll::Ready(Ok(())),
|
|
||||||
// capture error, all future readiness checks will fail
|
|
||||||
Poll::Ready(Err(e)) => {
|
|
||||||
*self.inner.error.borrow_mut() = Some(e.clone());
|
|
||||||
Poll::Ready(Err(e))
|
|
||||||
}
|
|
||||||
// start driver and elide all poll_ready calls until it is complete
|
|
||||||
Poll::Pending => {
|
|
||||||
let inner = self.inner.clone();
|
|
||||||
let (tx, rx) = oneshot::channel();
|
|
||||||
inner.driver_running.set(true);
|
|
||||||
inner.driver_stop.set(Some(tx));
|
|
||||||
|
|
||||||
ntex_rt::spawn(async move {
|
|
||||||
log::trace!("SharedService driver has started");
|
|
||||||
let service_ready = poll_fn(|cx| inner.service.poll_ready(cx));
|
|
||||||
let clients_gone = rx;
|
|
||||||
let result = select(service_ready, clients_gone).await;
|
|
||||||
if let Either::Left(result) = result {
|
|
||||||
log::trace!(
|
|
||||||
"SharedService driver completed, errored: {}",
|
|
||||||
result.is_err()
|
|
||||||
);
|
|
||||||
if let Err(e) = result {
|
|
||||||
inner.error.borrow_mut().replace(e);
|
|
||||||
}
|
|
||||||
inner.driver_running.set(false);
|
|
||||||
inner.driver_stop.set(None);
|
|
||||||
inner.ready.notify();
|
|
||||||
} else {
|
|
||||||
log::trace!("SharedService driver task stopped because all clients are gone");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// register waker to be notified, regardless of any previous notification
|
|
||||||
let _ = self.readiness.poll_ready(cx);
|
|
||||||
|
|
||||||
Poll::Pending
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn poll_shutdown(&self, cx: &mut Context<'_>) -> Poll<()> {
|
|
||||||
self.inner.service.poll_shutdown(cx)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn call(&self, req: R) -> Self::Future<'_> {
|
|
||||||
self.inner.service.call(req)
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -2,11 +2,9 @@
|
||||||
//!
|
//!
|
||||||
//! If the response does not complete within the specified timeout, the response
|
//! If the response does not complete within the specified timeout, the response
|
||||||
//! will be aborted.
|
//! will be aborted.
|
||||||
use std::{
|
use std::{fmt, future::Future, marker, pin::Pin, task::Context, task::Poll};
|
||||||
fmt, future::Future, marker, marker::PhantomData, pin::Pin, task::Context, task::Poll,
|
|
||||||
};
|
|
||||||
|
|
||||||
use ntex_service::{IntoService, Middleware, Service};
|
use ntex_service::{Ctx, IntoService, Middleware, Service, ServiceCall};
|
||||||
|
|
||||||
use crate::future::Either;
|
use crate::future::Either;
|
||||||
use crate::time::{sleep, Millis, Sleep};
|
use crate::time::{sleep, Millis, Sleep};
|
||||||
|
@ -127,17 +125,17 @@ where
|
||||||
type Error = TimeoutError<S::Error>;
|
type Error = TimeoutError<S::Error>;
|
||||||
type Future<'f> = Either<TimeoutServiceResponse<'f, S, R>, TimeoutServiceResponse2<'f, S, R>> where Self: 'f, R: 'f;
|
type Future<'f> = Either<TimeoutServiceResponse<'f, S, R>, TimeoutServiceResponse2<'f, S, R>> where Self: 'f, R: 'f;
|
||||||
|
|
||||||
fn call(&self, request: R) -> Self::Future<'_> {
|
fn call<'a>(&'a self, request: R, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
if self.timeout.is_zero() {
|
if self.timeout.is_zero() {
|
||||||
Either::Right(TimeoutServiceResponse2 {
|
Either::Right(TimeoutServiceResponse2 {
|
||||||
fut: self.service.call(request),
|
fut: ctx.call(&self.service, request),
|
||||||
_t: PhantomData,
|
_t: marker::PhantomData,
|
||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
Either::Left(TimeoutServiceResponse {
|
Either::Left(TimeoutServiceResponse {
|
||||||
fut: self.service.call(request),
|
fut: ctx.call(&self.service, request),
|
||||||
sleep: sleep(self.timeout),
|
sleep: sleep(self.timeout),
|
||||||
_t: PhantomData,
|
_t: marker::PhantomData,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -149,14 +147,14 @@ where
|
||||||
pin_project_lite::pin_project! {
|
pin_project_lite::pin_project! {
|
||||||
/// `TimeoutService` response future
|
/// `TimeoutService` response future
|
||||||
#[doc(hidden)]
|
#[doc(hidden)]
|
||||||
#[derive(Debug)]
|
#[must_use = "futures do nothing unless polled"]
|
||||||
pub struct TimeoutServiceResponse<'f, T: Service<R>, R>
|
pub struct TimeoutServiceResponse<'f, T: Service<R>, R>
|
||||||
where T: 'f, R: 'f,
|
where T: 'f, R: 'f,
|
||||||
{
|
{
|
||||||
#[pin]
|
#[pin]
|
||||||
fut: T::Future<'f>,
|
fut: ServiceCall<'f, T, R>,
|
||||||
sleep: Sleep,
|
sleep: Sleep,
|
||||||
_t: PhantomData<R>
|
_t: marker::PhantomData<R>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -187,13 +185,13 @@ where
|
||||||
pin_project_lite::pin_project! {
|
pin_project_lite::pin_project! {
|
||||||
/// `TimeoutService` response future
|
/// `TimeoutService` response future
|
||||||
#[doc(hidden)]
|
#[doc(hidden)]
|
||||||
#[derive(Debug)]
|
#[must_use = "futures do nothing unless polled"]
|
||||||
pub struct TimeoutServiceResponse2<'f, T: Service<R>, R>
|
pub struct TimeoutServiceResponse2<'f, T: Service<R>, R>
|
||||||
where T: 'f, R: 'f,
|
where T: 'f, R: 'f,
|
||||||
{
|
{
|
||||||
#[pin]
|
#[pin]
|
||||||
fut: T::Future<'f>,
|
fut: ServiceCall<'f, T, R>,
|
||||||
_t: PhantomData<R>,
|
_t: marker::PhantomData<R>,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -216,7 +214,7 @@ where
|
||||||
mod tests {
|
mod tests {
|
||||||
use std::{fmt, time::Duration};
|
use std::{fmt, time::Duration};
|
||||||
|
|
||||||
use ntex_service::{apply, fn_factory, Service, ServiceFactory};
|
use ntex_service::{apply, fn_factory, Container, Service, ServiceFactory};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::future::{lazy, BoxFuture};
|
use crate::future::{lazy, BoxFuture};
|
||||||
|
@ -238,7 +236,7 @@ mod tests {
|
||||||
type Error = SrvError;
|
type Error = SrvError;
|
||||||
type Future<'f> = BoxFuture<'f, Result<(), SrvError>>;
|
type Future<'f> = BoxFuture<'f, Result<(), SrvError>>;
|
||||||
|
|
||||||
fn call(&self, _: ()) -> Self::Future<'_> {
|
fn call<'a>(&'a self, _: (), _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
let fut = crate::time::sleep(self.0);
|
let fut = crate::time::sleep(self.0);
|
||||||
Box::pin(async move {
|
Box::pin(async move {
|
||||||
fut.await;
|
fut.await;
|
||||||
|
@ -252,7 +250,9 @@ mod tests {
|
||||||
let resolution = Duration::from_millis(100);
|
let resolution = Duration::from_millis(100);
|
||||||
let wait_time = Duration::from_millis(50);
|
let wait_time = Duration::from_millis(50);
|
||||||
|
|
||||||
let timeout = TimeoutService::new(resolution, SleepService(wait_time)).clone();
|
let timeout = Container::new(
|
||||||
|
TimeoutService::new(resolution, SleepService(wait_time)).clone(),
|
||||||
|
);
|
||||||
assert_eq!(timeout.call(()).await, Ok(()));
|
assert_eq!(timeout.call(()).await, Ok(()));
|
||||||
assert!(lazy(|cx| timeout.poll_ready(cx)).await.is_ready());
|
assert!(lazy(|cx| timeout.poll_ready(cx)).await.is_ready());
|
||||||
assert!(lazy(|cx| timeout.poll_shutdown(cx)).await.is_ready());
|
assert!(lazy(|cx| timeout.poll_shutdown(cx)).await.is_ready());
|
||||||
|
@ -263,7 +263,8 @@ mod tests {
|
||||||
let wait_time = Duration::from_millis(50);
|
let wait_time = Duration::from_millis(50);
|
||||||
let resolution = Duration::from_millis(0);
|
let resolution = Duration::from_millis(0);
|
||||||
|
|
||||||
let timeout = TimeoutService::new(resolution, SleepService(wait_time));
|
let timeout =
|
||||||
|
Container::new(TimeoutService::new(resolution, SleepService(wait_time)));
|
||||||
assert_eq!(timeout.call(()).await, Ok(()));
|
assert_eq!(timeout.call(()).await, Ok(()));
|
||||||
assert!(lazy(|cx| timeout.poll_ready(cx)).await.is_ready());
|
assert!(lazy(|cx| timeout.poll_ready(cx)).await.is_ready());
|
||||||
}
|
}
|
||||||
|
@ -273,7 +274,8 @@ mod tests {
|
||||||
let resolution = Duration::from_millis(100);
|
let resolution = Duration::from_millis(100);
|
||||||
let wait_time = Duration::from_millis(500);
|
let wait_time = Duration::from_millis(500);
|
||||||
|
|
||||||
let timeout = TimeoutService::new(resolution, SleepService(wait_time));
|
let timeout =
|
||||||
|
Container::new(TimeoutService::new(resolution, SleepService(wait_time)));
|
||||||
assert_eq!(timeout.call(()).await, Err(TimeoutError::Timeout));
|
assert_eq!(timeout.call(()).await, Err(TimeoutError::Timeout));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -287,7 +289,7 @@ mod tests {
|
||||||
Timeout::new(resolution).clone(),
|
Timeout::new(resolution).clone(),
|
||||||
fn_factory(|| async { Ok::<_, ()>(SleepService(wait_time)) }),
|
fn_factory(|| async { Ok::<_, ()>(SleepService(wait_time)) }),
|
||||||
);
|
);
|
||||||
let srv = timeout.create(&()).await.unwrap();
|
let srv = timeout.container(&()).await.unwrap();
|
||||||
|
|
||||||
let res = srv.call(()).await.unwrap_err();
|
let res = srv.call(()).await.unwrap_err();
|
||||||
assert_eq!(res, TimeoutError::Timeout);
|
assert_eq!(res, TimeoutError::Timeout);
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
//! Contains `Variant` service and related types and functions.
|
//! Contains `Variant` service and related types and functions.
|
||||||
use std::{future::Future, marker::PhantomData, pin::Pin, task::Context, task::Poll};
|
use std::{future::Future, marker::PhantomData, pin::Pin, task::Context, task::Poll};
|
||||||
|
|
||||||
use ntex_service::{IntoServiceFactory, Service, ServiceFactory};
|
use ntex_service::{Ctx, IntoServiceFactory, Service, ServiceCall, ServiceFactory};
|
||||||
|
|
||||||
/// Construct `Variant` service factory.
|
/// Construct `Variant` service factory.
|
||||||
///
|
///
|
||||||
|
@ -103,7 +103,8 @@ macro_rules! variant_impl ({$mod_name:ident, $enum_type:ident, $srv_type:ident,
|
||||||
{
|
{
|
||||||
type Response = V1::Response;
|
type Response = V1::Response;
|
||||||
type Error = V1::Error;
|
type Error = V1::Error;
|
||||||
type Future<'f> = $mod_name::ServiceResponse<V1::Future<'f>, $($T::Future<'f>),+> where Self: 'f, V1: 'f;
|
type Future<'f> = $mod_name::ServiceResponse<
|
||||||
|
ServiceCall<'f, V1, V1R>, $(ServiceCall<'f, $T, $R>),+> where Self: 'f, V1: 'f;
|
||||||
|
|
||||||
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
||||||
let mut ready = self.V1.poll_ready(cx)?.is_ready();
|
let mut ready = self.V1.poll_ready(cx)?.is_ready();
|
||||||
|
@ -127,10 +128,11 @@ macro_rules! variant_impl ({$mod_name:ident, $enum_type:ident, $srv_type:ident,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, req: $enum_type<V1R, $($R,)+>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: $enum_type<V1R, $($R,)+>, ctx: Ctx<'a, Self>) -> Self::Future<'a>
|
||||||
|
{
|
||||||
match req {
|
match req {
|
||||||
$enum_type::V1(req) => $mod_name::ServiceResponse::V1 { fut: self.V1.call(req) },
|
$enum_type::V1(req) => $mod_name::ServiceResponse::V1 { fut: ctx.call(&self.V1, req) },
|
||||||
$($enum_type::$T(req) => $mod_name::ServiceResponse::$T { fut: self.$T.call(req) },)+
|
$($enum_type::$T(req) => $mod_name::ServiceResponse::$T { fut: ctx.call(&self.$T, req) },)+
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -319,7 +321,7 @@ mod tests {
|
||||||
Poll::Ready(())
|
Poll::Ready(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, _: ()) -> Self::Future<'_> {
|
fn call<'a>(&'a self, _: (), _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
Ready::<_, ()>::Ok(1)
|
Ready::<_, ()>::Ok(1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -340,7 +342,7 @@ mod tests {
|
||||||
Poll::Ready(())
|
Poll::Ready(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, _: ()) -> Self::Future<'_> {
|
fn call<'a>(&'a self, _: (), _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
Ready::<_, ()>::Ok(2)
|
Ready::<_, ()>::Ok(2)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -352,7 +354,7 @@ mod tests {
|
||||||
.clone()
|
.clone()
|
||||||
.v3(fn_factory(|| async { Ok::<_, ()>(Srv2) }))
|
.v3(fn_factory(|| async { Ok::<_, ()>(Srv2) }))
|
||||||
.clone();
|
.clone();
|
||||||
let service = factory.create(&()).await.unwrap().clone();
|
let service = factory.container(&()).await.unwrap().clone();
|
||||||
|
|
||||||
assert!(lazy(|cx| service.poll_ready(cx)).await.is_ready());
|
assert!(lazy(|cx| service.poll_ready(cx)).await.is_ready());
|
||||||
assert!(lazy(|cx| service.poll_shutdown(cx)).await.is_ready());
|
assert!(lazy(|cx| service.poll_shutdown(cx)).await.is_ready());
|
||||||
|
|
|
@ -711,7 +711,7 @@ impl Future for LowresTimerDriver {
|
||||||
flags.remove(Flags::LOWRES_TIMER);
|
flags.remove(Flags::LOWRES_TIMER);
|
||||||
self.0.flags.set(flags);
|
self.0.flags.set(flags);
|
||||||
}
|
}
|
||||||
return Poll::Pending;
|
Poll::Pending
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex"
|
name = "ntex"
|
||||||
version = "0.6.7"
|
version = "0.7.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "Framework for composable network services"
|
description = "Framework for composable network services"
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
|
@ -49,20 +49,20 @@ async-std = ["ntex-rt/async-std", "ntex-async-std", "ntex-connect/async-std"]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-codec = "0.6.2"
|
ntex-codec = "0.6.2"
|
||||||
ntex-connect = "0.2.1"
|
ntex-connect = "0.3.0"
|
||||||
ntex-http = "0.1.9"
|
ntex-http = "0.1.9"
|
||||||
ntex-router = "0.5.1"
|
ntex-router = "0.5.1"
|
||||||
ntex-service = "1.0.2"
|
ntex-service = "1.2.0"
|
||||||
ntex-macros = "0.1.3"
|
ntex-macros = "0.1.3"
|
||||||
ntex-util = "0.2.3"
|
ntex-util = "0.3.0"
|
||||||
ntex-bytes = "0.1.19"
|
ntex-bytes = "0.1.19"
|
||||||
ntex-h2 = "0.2.3"
|
ntex-h2 = "0.3.0"
|
||||||
ntex-rt = "0.4.9"
|
ntex-rt = "0.4.9"
|
||||||
ntex-io = "0.2.10"
|
ntex-io = "0.3.0"
|
||||||
ntex-tls = "0.2.4"
|
ntex-tls = "0.3.0"
|
||||||
ntex-tokio = { version = "0.2.3", optional = true }
|
ntex-tokio = { version = "0.3.0", optional = true }
|
||||||
ntex-glommio = { version = "0.2.4", optional = true }
|
ntex-glommio = { version = "0.3.0", optional = true }
|
||||||
ntex-async-std = { version = "0.2.1", optional = true }
|
ntex-async-std = { version = "0.3.0", optional = true }
|
||||||
|
|
||||||
async-oneshot = "0.5.0"
|
async-oneshot = "0.5.0"
|
||||||
async-channel = "1.8.0"
|
async-channel = "1.8.0"
|
||||||
|
@ -94,8 +94,8 @@ coo-kie = { version = "0.17", package = "cookie", optional = true }
|
||||||
tls-openssl = { version="0.10", package = "openssl", optional = true }
|
tls-openssl = { version="0.10", package = "openssl", optional = true }
|
||||||
|
|
||||||
# rustls
|
# rustls
|
||||||
tls-rustls = { version = "0.20", package = "rustls", optional = true }
|
tls-rustls = { version = "0.21", package = "rustls", optional = true }
|
||||||
webpki-roots = { version = "0.22", optional = true }
|
webpki-roots = { version = "0.23", optional = true }
|
||||||
|
|
||||||
# compression
|
# compression
|
||||||
brotli2 = { version="0.3.2", optional = true }
|
brotli2 = { version="0.3.2", optional = true }
|
||||||
|
@ -107,6 +107,6 @@ rand = "0.8"
|
||||||
time = "0.3"
|
time = "0.3"
|
||||||
futures-util = "0.3"
|
futures-util = "0.3"
|
||||||
tls-openssl = { version="0.10", package = "openssl" }
|
tls-openssl = { version="0.10", package = "openssl" }
|
||||||
tls-rustls = { version = "0.20", package="rustls", features = ["dangerous_configuration"] }
|
tls-rustls = { version = "0.21", package="rustls", features = ["dangerous_configuration"] }
|
||||||
rustls-pemfile = { version = "1.0" }
|
rustls-pemfile = { version = "1.0" }
|
||||||
webpki-roots = { version = "0.22" }
|
webpki-roots = { version = "0.23" }
|
||||||
|
|
|
@ -55,6 +55,7 @@ impl<T: MessageBody> MessageBody for Box<T> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
pub enum ResponseBody<B> {
|
pub enum ResponseBody<B> {
|
||||||
Body(B),
|
Body(B),
|
||||||
Other(Body),
|
Other(Body),
|
||||||
|
|
|
@ -36,7 +36,7 @@ impl ClientBuilder {
|
||||||
config: ClientConfig {
|
config: ClientConfig {
|
||||||
headers: HeaderMap::new(),
|
headers: HeaderMap::new(),
|
||||||
timeout: Millis(5_000),
|
timeout: Millis(5_000),
|
||||||
connector: Box::new(ConnectorWrapper(Connector::default().finish())),
|
connector: Box::new(ConnectorWrapper(Connector::default().finish().into())),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -46,7 +46,7 @@ impl ClientBuilder {
|
||||||
where
|
where
|
||||||
T: Service<Connect, Response = Connection, Error = ConnectError> + 'static,
|
T: Service<Connect, Response = Connection, Error = ConnectError> + 'static,
|
||||||
{
|
{
|
||||||
self.config.connector = Box::new(ConnectorWrapper(connector));
|
self.config.connector = Box::new(ConnectorWrapper(connector.into()));
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,13 +1,13 @@
|
||||||
use std::net;
|
use std::net;
|
||||||
|
|
||||||
use crate::http::{body::Body, RequestHeadType};
|
use crate::http::{body::Body, RequestHeadType};
|
||||||
use crate::{service::Service, util::BoxFuture};
|
use crate::{service::Container, service::Service, util::BoxFuture};
|
||||||
|
|
||||||
use super::error::{ConnectError, SendRequestError};
|
use super::error::{ConnectError, SendRequestError};
|
||||||
use super::response::ClientResponse;
|
use super::response::ClientResponse;
|
||||||
use super::{Connect as ClientConnect, Connection};
|
use super::{Connect as ClientConnect, Connection};
|
||||||
|
|
||||||
pub(super) struct ConnectorWrapper<T>(pub(crate) T);
|
pub(super) struct ConnectorWrapper<T>(pub(crate) Container<T>);
|
||||||
|
|
||||||
pub(super) trait Connect {
|
pub(super) trait Connect {
|
||||||
fn send_request(
|
fn send_request(
|
||||||
|
|
|
@ -3,11 +3,9 @@ use std::{task::Context, task::Poll, time::Duration};
|
||||||
use ntex_h2::{self as h2};
|
use ntex_h2::{self as h2};
|
||||||
|
|
||||||
use crate::connect::{Connect as TcpConnect, Connector as TcpConnector};
|
use crate::connect::{Connect as TcpConnect, Connector as TcpConnector};
|
||||||
use crate::service::{apply_fn, boxed, Service};
|
use crate::service::{apply_fn, boxed, Ctx, Service, ServiceCall};
|
||||||
use crate::time::{Millis, Seconds};
|
use crate::time::{Millis, Seconds};
|
||||||
use crate::util::{
|
use crate::util::{timeout::TimeoutError, timeout::TimeoutService, Either, Ready};
|
||||||
shared::SharedService, timeout::TimeoutError, timeout::TimeoutService, Either, Ready,
|
|
||||||
};
|
|
||||||
use crate::{http::Uri, io::IoBoxed};
|
use crate::{http::Uri, io::IoBoxed};
|
||||||
|
|
||||||
use super::{connection::Connection, error::ConnectError, pool::ConnectionPool, Connect};
|
use super::{connection::Connection, error::ConnectError, pool::ConnectionPool, Connect};
|
||||||
|
@ -216,7 +214,7 @@ impl Connector {
|
||||||
/// its combinator chain.
|
/// its combinator chain.
|
||||||
pub fn finish(
|
pub fn finish(
|
||||||
self,
|
self,
|
||||||
) -> impl Service<Connect, Response = Connection, Error = ConnectError> + Clone {
|
) -> impl Service<Connect, Response = Connection, Error = ConnectError> {
|
||||||
let tcp_service = connector(self.connector, self.timeout, self.disconnect_timeout);
|
let tcp_service = connector(self.connector, self.timeout, self.disconnect_timeout);
|
||||||
|
|
||||||
let ssl_pool = if let Some(ssl_connector) = self.ssl_connector {
|
let ssl_pool = if let Some(ssl_connector) = self.ssl_connector {
|
||||||
|
@ -233,7 +231,7 @@ impl Connector {
|
||||||
None
|
None
|
||||||
};
|
};
|
||||||
|
|
||||||
SharedService::new(InnerConnector {
|
InnerConnector {
|
||||||
tcp_pool: ConnectionPool::new(
|
tcp_pool: ConnectionPool::new(
|
||||||
tcp_service,
|
tcp_service,
|
||||||
self.conn_lifetime,
|
self.conn_lifetime,
|
||||||
|
@ -243,7 +241,7 @@ impl Connector {
|
||||||
self.h2config.clone(),
|
self.h2config.clone(),
|
||||||
),
|
),
|
||||||
ssl_pool,
|
ssl_pool,
|
||||||
})
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -283,7 +281,7 @@ where
|
||||||
type Response = <ConnectionPool<T> as Service<Connect>>::Response;
|
type Response = <ConnectionPool<T> as Service<Connect>>::Response;
|
||||||
type Error = ConnectError;
|
type Error = ConnectError;
|
||||||
type Future<'f> = Either<
|
type Future<'f> = Either<
|
||||||
<ConnectionPool<T> as Service<Connect>>::Future<'f>,
|
ServiceCall<'f, ConnectionPool<T>, Connect>,
|
||||||
Ready<Self::Response, Self::Error>,
|
Ready<Self::Response, Self::Error>,
|
||||||
>;
|
>;
|
||||||
|
|
||||||
|
@ -317,16 +315,16 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, req: Connect) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: Connect, ctx: Ctx<'a, Self>) -> Self::Future<'_> {
|
||||||
match req.uri.scheme_str() {
|
match req.uri.scheme_str() {
|
||||||
Some("https") | Some("wss") => {
|
Some("https") | Some("wss") => {
|
||||||
if let Some(ref conn) = self.ssl_pool {
|
if let Some(ref conn) = self.ssl_pool {
|
||||||
Either::Left(conn.call(req))
|
Either::Left(ctx.call(conn, req))
|
||||||
} else {
|
} else {
|
||||||
Either::Right(Ready::Err(ConnectError::SslIsNotSupported))
|
Either::Right(Ready::Err(ConnectError::SslIsNotSupported))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_ => Either::Left(self.tcp_pool.call(req)),
|
_ => Either::Left(ctx.call(&self.tcp_pool, req)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,7 +7,7 @@ use crate::http::header::{self, HeaderMap, HeaderValue};
|
||||||
use crate::http::message::{RequestHeadType, ResponseHead};
|
use crate::http::message::{RequestHeadType, ResponseHead};
|
||||||
use crate::http::{h2::payload, payload::Payload, Method, Version};
|
use crate::http::{h2::payload, payload::Payload, Method, Version};
|
||||||
use crate::util::{poll_fn, ByteString, Bytes, HashMap, Ready};
|
use crate::util::{poll_fn, ByteString, Bytes, HashMap, Ready};
|
||||||
use crate::{channel::oneshot, service::Service};
|
use crate::{channel::oneshot, service::Ctx, service::Service};
|
||||||
|
|
||||||
use super::error::SendRequestError;
|
use super::error::SendRequestError;
|
||||||
|
|
||||||
|
@ -226,7 +226,7 @@ impl Service<h2::Message> for H2PublishService {
|
||||||
type Error = &'static str;
|
type Error = &'static str;
|
||||||
type Future<'f> = Ready<Self::Response, Self::Error>;
|
type Future<'f> = Ready<Self::Response, Self::Error>;
|
||||||
|
|
||||||
fn call(&self, mut msg: h2::Message) -> Self::Future<'_> {
|
fn call<'a>(&'a self, mut msg: h2::Message, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
match msg.kind().take() {
|
match msg.kind().take() {
|
||||||
h2::MessageKind::Headers {
|
h2::MessageKind::Headers {
|
||||||
pseudo,
|
pseudo,
|
||||||
|
|
|
@ -82,7 +82,7 @@ pub(self) struct ClientConfig {
|
||||||
impl Default for Client {
|
impl Default for Client {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
Client(Rc::new(ClientConfig {
|
Client(Rc::new(ClientConfig {
|
||||||
connector: Box::new(ConnectorWrapper(Connector::default().finish())),
|
connector: Box::new(ConnectorWrapper(Connector::default().finish().into())),
|
||||||
headers: HeaderMap::new(),
|
headers: HeaderMap::new(),
|
||||||
timeout: Millis(5_000),
|
timeout: Millis(5_000),
|
||||||
}))
|
}))
|
||||||
|
|
|
@ -6,9 +6,10 @@ use ntex_h2::{self as h2};
|
||||||
|
|
||||||
use crate::http::uri::{Authority, Scheme, Uri};
|
use crate::http::uri::{Authority, Scheme, Uri};
|
||||||
use crate::io::{types::HttpProtocol, IoBoxed};
|
use crate::io::{types::HttpProtocol, IoBoxed};
|
||||||
|
use crate::service::{Container, Ctx, Service, ServiceCall};
|
||||||
use crate::time::{now, Millis};
|
use crate::time::{now, Millis};
|
||||||
use crate::util::{ready, BoxFuture, ByteString, HashMap, HashSet};
|
use crate::util::{ready, BoxFuture, ByteString, HashMap, HashSet};
|
||||||
use crate::{channel::pool, rt::spawn, service::Service, task::LocalWaker};
|
use crate::{channel::pool, rt::spawn, task::LocalWaker};
|
||||||
|
|
||||||
use super::connection::{Connection, ConnectionType};
|
use super::connection::{Connection, ConnectionType};
|
||||||
use super::h2proto::{H2Client, H2PublishService};
|
use super::h2proto::{H2Client, H2PublishService};
|
||||||
|
@ -43,7 +44,7 @@ struct AvailableConnection {
|
||||||
|
|
||||||
/// Connections pool
|
/// Connections pool
|
||||||
pub(super) struct ConnectionPool<T> {
|
pub(super) struct ConnectionPool<T> {
|
||||||
connector: Rc<T>,
|
connector: Container<T>,
|
||||||
inner: Rc<RefCell<Inner>>,
|
inner: Rc<RefCell<Inner>>,
|
||||||
waiters: Rc<RefCell<Waiters>>,
|
waiters: Rc<RefCell<Waiters>>,
|
||||||
}
|
}
|
||||||
|
@ -60,7 +61,7 @@ where
|
||||||
limit: usize,
|
limit: usize,
|
||||||
h2config: h2::Config,
|
h2config: h2::Config,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let connector = Rc::new(connector);
|
let connector = Container::new(connector);
|
||||||
let waiters = Rc::new(RefCell::new(Waiters {
|
let waiters = Rc::new(RefCell::new(Waiters {
|
||||||
waiters: HashMap::default(),
|
waiters: HashMap::default(),
|
||||||
pool: pool::new(),
|
pool: pool::new(),
|
||||||
|
@ -120,10 +121,8 @@ where
|
||||||
crate::forward_poll_ready!(connector);
|
crate::forward_poll_ready!(connector);
|
||||||
crate::forward_poll_shutdown!(connector);
|
crate::forward_poll_shutdown!(connector);
|
||||||
|
|
||||||
#[inline]
|
fn call<'a>(&'a self, req: Connect, _: Ctx<'a, Self>) -> Self::Future<'_> {
|
||||||
fn call(&self, req: Connect) -> Self::Future<'_> {
|
|
||||||
trace!("Get connection for {:?}", req.uri);
|
trace!("Get connection for {:?}", req.uri);
|
||||||
let connector = self.connector.clone();
|
|
||||||
let inner = self.inner.clone();
|
let inner = self.inner.clone();
|
||||||
let waiters = self.waiters.clone();
|
let waiters = self.waiters.clone();
|
||||||
|
|
||||||
|
@ -151,7 +150,7 @@ where
|
||||||
trace!("Connecting to {:?}", req.uri);
|
trace!("Connecting to {:?}", req.uri);
|
||||||
let uri = req.uri.clone();
|
let uri = req.uri.clone();
|
||||||
let (tx, rx) = waiters.borrow_mut().pool.channel();
|
let (tx, rx) = waiters.borrow_mut().pool.channel();
|
||||||
OpenConnection::spawn(key, tx, uri, inner, connector, req);
|
OpenConnection::spawn(key, tx, uri, inner, self.connector.clone(), req);
|
||||||
|
|
||||||
match rx.await {
|
match rx.await {
|
||||||
Err(_) => Err(ConnectError::Disconnected(None)),
|
Err(_) => Err(ConnectError::Disconnected(None)),
|
||||||
|
@ -308,7 +307,7 @@ impl Inner {
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ConnectionPoolSupport<T> {
|
struct ConnectionPoolSupport<T> {
|
||||||
connector: Rc<T>,
|
connector: Container<T>,
|
||||||
inner: Rc<RefCell<Inner>>,
|
inner: Rc<RefCell<Inner>>,
|
||||||
waiters: Rc<RefCell<Waiters>>,
|
waiters: Rc<RefCell<Waiters>>,
|
||||||
}
|
}
|
||||||
|
@ -391,7 +390,7 @@ pin_project_lite::pin_project! {
|
||||||
{
|
{
|
||||||
key: Key,
|
key: Key,
|
||||||
#[pin]
|
#[pin]
|
||||||
fut: T::Future<'f>,
|
fut: ServiceCall<'f, T, Connect>,
|
||||||
uri: Uri,
|
uri: Uri,
|
||||||
tx: Option<Waiter>,
|
tx: Option<Waiter>,
|
||||||
guard: Option<OpenGuard>,
|
guard: Option<OpenGuard>,
|
||||||
|
@ -409,11 +408,12 @@ where
|
||||||
tx: Waiter,
|
tx: Waiter,
|
||||||
uri: Uri,
|
uri: Uri,
|
||||||
inner: Rc<RefCell<Inner>>,
|
inner: Rc<RefCell<Inner>>,
|
||||||
connector: Rc<T>,
|
connector: Container<T>,
|
||||||
msg: Connect,
|
msg: Connect,
|
||||||
) {
|
) {
|
||||||
let disconnect_timeout = inner.borrow().disconnect_timeout;
|
let disconnect_timeout = inner.borrow().disconnect_timeout;
|
||||||
|
|
||||||
|
#[allow(clippy::redundant_async_block)]
|
||||||
spawn(async move {
|
spawn(async move {
|
||||||
OpenConnection::<T> {
|
OpenConnection::<T> {
|
||||||
fut: connector.call(msg),
|
fut: connector.call(msg),
|
||||||
|
@ -629,19 +629,21 @@ mod tests {
|
||||||
let store = Rc::new(RefCell::new(Vec::new()));
|
let store = Rc::new(RefCell::new(Vec::new()));
|
||||||
let store2 = store.clone();
|
let store2 = store.clone();
|
||||||
|
|
||||||
let pool = ConnectionPool::new(
|
let pool = Container::new(
|
||||||
fn_service(move |req| {
|
ConnectionPool::new(
|
||||||
let (client, server) = Io::create();
|
fn_service(move |req| {
|
||||||
store2.borrow_mut().push((req, server));
|
let (client, server) = Io::create();
|
||||||
Box::pin(async move { Ok(IoBoxed::from(nio::Io::new(client))) })
|
store2.borrow_mut().push((req, server));
|
||||||
}),
|
Box::pin(async move { Ok(IoBoxed::from(nio::Io::new(client))) })
|
||||||
Duration::from_secs(10),
|
}),
|
||||||
Duration::from_secs(10),
|
Duration::from_secs(10),
|
||||||
Millis::ZERO,
|
Duration::from_secs(10),
|
||||||
1,
|
Millis::ZERO,
|
||||||
h2::Config::client(),
|
1,
|
||||||
)
|
h2::Config::client(),
|
||||||
.clone();
|
)
|
||||||
|
.clone(),
|
||||||
|
);
|
||||||
|
|
||||||
// uri must contain authority
|
// uri must contain authority
|
||||||
let req = Connect {
|
let req = Connect {
|
||||||
|
|
|
@ -3,8 +3,9 @@ use std::{cell::Cell, ptr::copy_nonoverlapping, rc::Rc, time, time::Duration};
|
||||||
use ntex_h2::{self as h2};
|
use ntex_h2::{self as h2};
|
||||||
|
|
||||||
use crate::http::{Request, Response};
|
use crate::http::{Request, Response};
|
||||||
|
use crate::service::{boxed::BoxService, Container};
|
||||||
use crate::time::{sleep, Millis, Seconds};
|
use crate::time::{sleep, Millis, Seconds};
|
||||||
use crate::{io::IoRef, service::boxed::BoxService, util::BytesMut};
|
use crate::{io::IoRef, util::BytesMut};
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
|
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
|
||||||
/// Server keep-alive setting
|
/// Server keep-alive setting
|
||||||
|
@ -101,16 +102,16 @@ impl ServiceConfig {
|
||||||
pub(super) type OnRequest = BoxService<(Request, IoRef), Request, Response>;
|
pub(super) type OnRequest = BoxService<(Request, IoRef), Request, Response>;
|
||||||
|
|
||||||
pub(super) struct DispatcherConfig<S, X, U> {
|
pub(super) struct DispatcherConfig<S, X, U> {
|
||||||
pub(super) service: S,
|
pub(super) service: Container<S>,
|
||||||
pub(super) expect: X,
|
pub(super) expect: Container<X>,
|
||||||
pub(super) upgrade: Option<U>,
|
pub(super) upgrade: Option<Container<U>>,
|
||||||
pub(super) keep_alive: Duration,
|
pub(super) keep_alive: Duration,
|
||||||
pub(super) client_timeout: Duration,
|
pub(super) client_timeout: Duration,
|
||||||
pub(super) client_disconnect: Seconds,
|
pub(super) client_disconnect: Seconds,
|
||||||
pub(super) h2config: h2::Config,
|
pub(super) h2config: h2::Config,
|
||||||
pub(super) ka_enabled: bool,
|
pub(super) ka_enabled: bool,
|
||||||
pub(super) timer: DateService,
|
pub(super) timer: DateService,
|
||||||
pub(super) on_request: Option<OnRequest>,
|
pub(super) on_request: Option<Container<OnRequest>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S, X, U> DispatcherConfig<S, X, U> {
|
impl<S, X, U> DispatcherConfig<S, X, U> {
|
||||||
|
@ -122,10 +123,10 @@ impl<S, X, U> DispatcherConfig<S, X, U> {
|
||||||
on_request: Option<OnRequest>,
|
on_request: Option<OnRequest>,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
DispatcherConfig {
|
DispatcherConfig {
|
||||||
service,
|
service: service.into(),
|
||||||
expect,
|
expect: expect.into(),
|
||||||
upgrade,
|
upgrade: upgrade.map(|v| v.into()),
|
||||||
on_request,
|
on_request: on_request.map(|v| v.into()),
|
||||||
keep_alive: Duration::from(cfg.0.keep_alive),
|
keep_alive: Duration::from(cfg.0.keep_alive),
|
||||||
client_timeout: Duration::from(cfg.0.client_timeout),
|
client_timeout: Duration::from(cfg.0.client_timeout),
|
||||||
client_disconnect: cfg.0.client_disconnect,
|
client_disconnect: cfg.0.client_disconnect,
|
||||||
|
|
|
@ -2,8 +2,9 @@
|
||||||
use std::task::{Context, Poll};
|
use std::task::{Context, Poll};
|
||||||
use std::{cell::RefCell, error::Error, future::Future, io, marker, mem, pin::Pin, rc::Rc};
|
use std::{cell::RefCell, error::Error, future::Future, io, marker, mem, pin::Pin, rc::Rc};
|
||||||
|
|
||||||
use crate::io::{Filter, Io, IoBoxed, IoStatusUpdate, RecvError};
|
use crate::io::{Filter, Io, IoBoxed, IoRef, IoStatusUpdate, RecvError};
|
||||||
use crate::{service::Service, util::ready, util::BoxFuture, util::Bytes};
|
use crate::service::{Container, Service, ServiceCall};
|
||||||
|
use crate::util::{ready, Bytes};
|
||||||
|
|
||||||
use crate::http;
|
use crate::http;
|
||||||
use crate::http::body::{BodySize, MessageBody, ResponseBody};
|
use crate::http::body::{BodySize, MessageBody, ResponseBody};
|
||||||
|
@ -46,7 +47,7 @@ pin_project_lite::pin_project! {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(thiserror::Error)]
|
#[derive(Debug, thiserror::Error)]
|
||||||
enum State<B> {
|
enum State<B> {
|
||||||
#[error("State::Call")]
|
#[error("State::Call")]
|
||||||
Call,
|
Call,
|
||||||
|
@ -77,10 +78,10 @@ pin_project_lite::pin_project! {
|
||||||
where S: 'static, X: 'static
|
where S: 'static, X: 'static
|
||||||
{
|
{
|
||||||
None,
|
None,
|
||||||
Service { #[pin] fut: S::Future<'static> },
|
Service { #[pin] fut: ServiceCall<'static, S, Request> },
|
||||||
ServiceUpgrade { #[pin] fut: S::Future<'static> },
|
ServiceUpgrade { #[pin] fut: ServiceCall<'static, S, Request> },
|
||||||
Expect { #[pin] fut: X::Future<'static> },
|
Expect { #[pin] fut: ServiceCall<'static, X, Request> },
|
||||||
Filter { fut: BoxFuture<'static, Result<Request, Response>> }
|
Filter { fut: ServiceCall<'static, OnRequest, (Request, IoRef)> }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -485,7 +486,7 @@ where
|
||||||
st
|
st
|
||||||
}
|
}
|
||||||
|
|
||||||
fn service_filter(&self, req: Request, f: &OnRequest) -> CallState<S, X> {
|
fn service_filter(&self, req: Request, f: &Container<OnRequest>) -> CallState<S, X> {
|
||||||
// Handle filter fut
|
// Handle filter fut
|
||||||
let fut = f.call((req, self.io.get_ref()));
|
let fut = f.call((req, self.io.get_ref()));
|
||||||
let st = CallState::Filter {
|
let st = CallState::Filter {
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use std::io;
|
use std::io;
|
||||||
|
|
||||||
use crate::http::request::Request;
|
use crate::service::{Ctx, Service, ServiceFactory};
|
||||||
use crate::{service::Service, service::ServiceFactory, util::Ready};
|
use crate::{http::request::Request, util::Ready};
|
||||||
|
|
||||||
pub struct ExpectHandler;
|
pub struct ExpectHandler;
|
||||||
|
|
||||||
|
@ -24,7 +24,7 @@ impl Service<Request> for ExpectHandler {
|
||||||
type Future<'f> = Ready<Self::Response, Self::Error>;
|
type Future<'f> = Ready<Self::Response, Self::Error>;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: Request) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: Request, _: Ctx<'a, Self>) -> Self::Future<'_> {
|
||||||
Ready::Ok(req)
|
Ready::Ok(req)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,7 +5,7 @@ use crate::http::config::{DispatcherConfig, OnRequest, ServiceConfig};
|
||||||
use crate::http::error::{DispatchError, ResponseError};
|
use crate::http::error::{DispatchError, ResponseError};
|
||||||
use crate::http::{request::Request, response::Response};
|
use crate::http::{request::Request, response::Response};
|
||||||
use crate::io::{types, Filter, Io};
|
use crate::io::{types, Filter, Io};
|
||||||
use crate::service::{IntoServiceFactory, Service, ServiceFactory};
|
use crate::service::{Ctx, IntoServiceFactory, Service, ServiceFactory};
|
||||||
use crate::{time::Millis, util::BoxFuture};
|
use crate::{time::Millis, util::BoxFuture};
|
||||||
|
|
||||||
use super::codec::Codec;
|
use super::codec::Codec;
|
||||||
|
@ -331,7 +331,7 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, io: Io<F>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, io: Io<F>, _: Ctx<'a, Self>) -> Self::Future<'_> {
|
||||||
log::trace!(
|
log::trace!(
|
||||||
"New http1 connection, peer address {:?}",
|
"New http1 connection, peer address {:?}",
|
||||||
io.query::<types::PeerAddr>().get()
|
io.query::<types::PeerAddr>().get()
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use std::{io, marker::PhantomData};
|
use std::{io, marker::PhantomData};
|
||||||
|
|
||||||
use crate::http::{h1::Codec, request::Request};
|
use crate::http::{h1::Codec, request::Request};
|
||||||
use crate::{io::Io, service::Service, service::ServiceFactory, util::Ready};
|
use crate::{io::Io, service::Ctx, service::Service, service::ServiceFactory, util::Ready};
|
||||||
|
|
||||||
pub struct UpgradeHandler<F>(PhantomData<F>);
|
pub struct UpgradeHandler<F>(PhantomData<F>);
|
||||||
|
|
||||||
|
@ -25,7 +25,11 @@ impl<F> Service<(Request, Io<F>, Codec)> for UpgradeHandler<F> {
|
||||||
type Future<'f> = Ready<Self::Response, Self::Error> where F: 'f;
|
type Future<'f> = Ready<Self::Response, Self::Error> where F: 'f;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, _: (Request, Io<F>, Codec)) -> Self::Future<'_> {
|
fn call<'a>(
|
||||||
|
&'a self,
|
||||||
|
_: (Request, Io<F>, Codec),
|
||||||
|
_: Ctx<'a, Self>,
|
||||||
|
) -> Self::Future<'a> {
|
||||||
unimplemented!()
|
unimplemented!()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,7 +10,7 @@ use crate::http::header::{self, HeaderMap, HeaderName, HeaderValue};
|
||||||
use crate::http::message::{CurrentIo, ResponseHead};
|
use crate::http::message::{CurrentIo, ResponseHead};
|
||||||
use crate::http::{DateService, Method, Request, Response, StatusCode, Uri, Version};
|
use crate::http::{DateService, Method, Request, Response, StatusCode, Uri, Version};
|
||||||
use crate::io::{types, Filter, Io, IoBoxed, IoRef};
|
use crate::io::{types, Filter, Io, IoBoxed, IoRef};
|
||||||
use crate::service::{IntoServiceFactory, Service, ServiceFactory};
|
use crate::service::{Ctx, IntoServiceFactory, Service, ServiceFactory};
|
||||||
use crate::util::{poll_fn, BoxFuture, Bytes, BytesMut, Either, HashMap, Ready};
|
use crate::util::{poll_fn, BoxFuture, Bytes, BytesMut, Either, HashMap, Ready};
|
||||||
|
|
||||||
use super::payload::{Payload, PayloadSender};
|
use super::payload::{Payload, PayloadSender};
|
||||||
|
@ -181,7 +181,7 @@ where
|
||||||
self.config.service.poll_shutdown(cx)
|
self.config.service.poll_shutdown(cx)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, io: Io<F>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, io: Io<F>, _: Ctx<'a, Self>) -> Self::Future<'_> {
|
||||||
log::trace!(
|
log::trace!(
|
||||||
"New http2 connection, peer address {:?}",
|
"New http2 connection, peer address {:?}",
|
||||||
io.query::<types::PeerAddr>().get()
|
io.query::<types::PeerAddr>().get()
|
||||||
|
@ -230,7 +230,11 @@ impl Service<h2::ControlMessage<H2Error>> for ControlService {
|
||||||
type Error = ();
|
type Error = ();
|
||||||
type Future<'f> = Ready<Self::Response, Self::Error>;
|
type Future<'f> = Ready<Self::Response, Self::Error>;
|
||||||
|
|
||||||
fn call(&self, msg: h2::ControlMessage<H2Error>) -> Self::Future<'_> {
|
fn call<'a>(
|
||||||
|
&'a self,
|
||||||
|
msg: h2::ControlMessage<H2Error>,
|
||||||
|
_: Ctx<'a, Self>,
|
||||||
|
) -> Self::Future<'a> {
|
||||||
log::trace!("Control message: {:?}", msg);
|
log::trace!("Control message: {:?}", msg);
|
||||||
Ready::Ok::<_, ()>(msg.ack())
|
Ready::Ok::<_, ()>(msg.ack())
|
||||||
}
|
}
|
||||||
|
@ -276,7 +280,7 @@ where
|
||||||
Ready<Self::Response, Self::Error>,
|
Ready<Self::Response, Self::Error>,
|
||||||
>;
|
>;
|
||||||
|
|
||||||
fn call(&self, mut msg: h2::Message) -> Self::Future<'_> {
|
fn call<'a>(&'a self, mut msg: h2::Message, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
let (io, pseudo, headers, eof, payload) = match msg.kind().take() {
|
let (io, pseudo, headers, eof, payload) = match msg.kind().take() {
|
||||||
h2::MessageKind::Headers {
|
h2::MessageKind::Headers {
|
||||||
pseudo,
|
pseudo,
|
||||||
|
|
|
@ -7,19 +7,15 @@ use crate::util::{poll_fn, Bytes, Stream};
|
||||||
pub type PayloadStream = Pin<Box<dyn Stream<Item = Result<Bytes, PayloadError>>>>;
|
pub type PayloadStream = Pin<Box<dyn Stream<Item = Result<Bytes, PayloadError>>>>;
|
||||||
|
|
||||||
/// Type represent streaming payload
|
/// Type represent streaming payload
|
||||||
|
#[derive(Default)]
|
||||||
pub enum Payload {
|
pub enum Payload {
|
||||||
|
#[default]
|
||||||
None,
|
None,
|
||||||
H1(h1::Payload),
|
H1(h1::Payload),
|
||||||
H2(h2::Payload),
|
H2(h2::Payload),
|
||||||
Stream(PayloadStream),
|
Stream(PayloadStream),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for Payload {
|
|
||||||
fn default() -> Self {
|
|
||||||
Payload::None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<h1::Payload> for Payload {
|
impl From<h1::Payload> for Payload {
|
||||||
fn from(v: h1::Payload) -> Self {
|
fn from(v: h1::Payload) -> Self {
|
||||||
Payload::H1(v)
|
Payload::H1(v)
|
||||||
|
|
|
@ -2,7 +2,7 @@ use std::task::{Context, Poll};
|
||||||
use std::{cell, error, fmt, future, marker, pin::Pin, rc::Rc};
|
use std::{cell, error, fmt, future, marker, pin::Pin, rc::Rc};
|
||||||
|
|
||||||
use crate::io::{types, Filter, Io};
|
use crate::io::{types, Filter, Io};
|
||||||
use crate::service::{IntoServiceFactory, Service, ServiceFactory};
|
use crate::service::{Ctx, IntoServiceFactory, Service, ServiceFactory};
|
||||||
use crate::time::{Millis, Seconds};
|
use crate::time::{Millis, Seconds};
|
||||||
use crate::util::BoxFuture;
|
use crate::util::BoxFuture;
|
||||||
|
|
||||||
|
@ -368,7 +368,7 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, io: Io<F>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, io: Io<F>, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
log::trace!(
|
log::trace!(
|
||||||
"New http connection, peer address {:?}",
|
"New http connection, peer address {:?}",
|
||||||
io.query::<types::PeerAddr>().get()
|
io.query::<types::PeerAddr>().get()
|
||||||
|
|
|
@ -38,12 +38,11 @@ pub mod web;
|
||||||
pub mod ws;
|
pub mod ws;
|
||||||
|
|
||||||
pub use self::service::{
|
pub use self::service::{
|
||||||
fn_service, into_service, pipeline, pipeline_factory, IntoService, IntoServiceFactory,
|
fn_service, into_service, pipeline, pipeline_factory, Container, Ctx, IntoService,
|
||||||
Middleware, Service, ServiceFactory,
|
IntoServiceFactory, Middleware, Service, ServiceCall, ServiceFactory,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub use ntex_util::channel;
|
pub use ntex_util::{channel, task};
|
||||||
pub use ntex_util::task;
|
|
||||||
|
|
||||||
pub mod codec {
|
pub mod codec {
|
||||||
//! Utilities for encoding and decoding frames.
|
//! Utilities for encoding and decoding frames.
|
||||||
|
|
|
@ -1,19 +1,15 @@
|
||||||
use std::{
|
use std::{cell::Cell, cell::RefCell, fmt, future::Future, io, marker, mem, net, rc::Rc};
|
||||||
cell::Cell, cell::RefCell, fmt, future::Future, io, marker::PhantomData, mem, net,
|
|
||||||
rc::Rc,
|
|
||||||
};
|
|
||||||
|
|
||||||
use log::error;
|
use log::error;
|
||||||
|
|
||||||
|
use crate::io::Io;
|
||||||
use crate::service::{self, boxed, ServiceFactory as NServiceFactory};
|
use crate::service::{self, boxed, ServiceFactory as NServiceFactory};
|
||||||
use crate::util::{BoxFuture, HashMap, Ready};
|
use crate::util::{BoxFuture, HashMap, PoolId, Ready};
|
||||||
use crate::{io::Io, util::PoolId};
|
|
||||||
|
|
||||||
use super::service::{
|
use super::service::{
|
||||||
BoxedServerService, InternalServiceFactory, ServerMessage, StreamService,
|
BoxedServerService, InternalServiceFactory, ServerMessage, StreamService,
|
||||||
};
|
};
|
||||||
use super::Token;
|
use super::{builder::bind_addr, counter::CounterGuard, Token};
|
||||||
use super::{builder::bind_addr, counter::CounterGuard};
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct Config(Rc<InnerServiceConfig>);
|
pub struct Config(Rc<InnerServiceConfig>);
|
||||||
|
@ -66,7 +62,7 @@ impl ServiceConfig {
|
||||||
not_configured();
|
not_configured();
|
||||||
Ready::Ok::<_, &'static str>(())
|
Ready::Ok::<_, &'static str>(())
|
||||||
},
|
},
|
||||||
_t: PhantomData,
|
_t: marker::PhantomData,
|
||||||
})),
|
})),
|
||||||
})))
|
})))
|
||||||
}
|
}
|
||||||
|
@ -95,7 +91,7 @@ impl ServiceConfig {
|
||||||
not_configured();
|
not_configured();
|
||||||
Ready::Ok::<_, &'static str>(())
|
Ready::Ok::<_, &'static str>(())
|
||||||
},
|
},
|
||||||
_t: PhantomData,
|
_t: marker::PhantomData,
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
inner.services.push((name.as_ref().to_string(), lst));
|
inner.services.push((name.as_ref().to_string(), lst));
|
||||||
|
@ -114,7 +110,10 @@ impl ServiceConfig {
|
||||||
E: fmt::Display + 'static,
|
E: fmt::Display + 'static,
|
||||||
{
|
{
|
||||||
self.0.borrow_mut().applied = true;
|
self.0.borrow_mut().applied = true;
|
||||||
self.0.borrow_mut().apply = Some(Box::new(ConfigWrapper { f, _t: PhantomData }));
|
self.0.borrow_mut().apply = Some(Box::new(ConfigWrapper {
|
||||||
|
f,
|
||||||
|
_t: marker::PhantomData,
|
||||||
|
}));
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -192,7 +191,7 @@ impl InternalServiceFactory for ConfiguredService {
|
||||||
let name = names.remove(&token).unwrap().0;
|
let name = names.remove(&token).unwrap().0;
|
||||||
res.push((
|
res.push((
|
||||||
token,
|
token,
|
||||||
boxed::rcservice(StreamService::new(
|
boxed::service(StreamService::new(
|
||||||
service::fn_service(move |_: Io| {
|
service::fn_service(move |_: Io| {
|
||||||
error!("Service {:?} is not configured", name);
|
error!("Service {:?} is not configured", name);
|
||||||
Ready::<_, ()>::Ok(())
|
Ready::<_, ()>::Ok(())
|
||||||
|
@ -215,7 +214,7 @@ pub(super) trait ServiceRuntimeConfiguration {
|
||||||
|
|
||||||
pub(super) struct ConfigWrapper<F, R, E> {
|
pub(super) struct ConfigWrapper<F, R, E> {
|
||||||
pub(super) f: F,
|
pub(super) f: F,
|
||||||
pub(super) _t: PhantomData<(R, E)>,
|
pub(super) _t: marker::PhantomData<(R, E)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
// SAFETY: we dont store R or E in ConfigWrapper
|
// SAFETY: we dont store R or E in ConfigWrapper
|
||||||
|
@ -230,7 +229,7 @@ where
|
||||||
fn clone(&self) -> Box<dyn ServiceRuntimeConfiguration + Send> {
|
fn clone(&self) -> Box<dyn ServiceRuntimeConfiguration + Send> {
|
||||||
Box::new(ConfigWrapper {
|
Box::new(ConfigWrapper {
|
||||||
f: self.f.clone(),
|
f: self.f.clone(),
|
||||||
_t: PhantomData,
|
_t: marker::PhantomData,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -304,7 +303,7 @@ impl ServiceRuntime {
|
||||||
let token = *token;
|
let token = *token;
|
||||||
inner.services.insert(
|
inner.services.insert(
|
||||||
token,
|
token,
|
||||||
boxed::rcfactory(ServiceFactory {
|
boxed::factory(ServiceFactory {
|
||||||
pool,
|
pool,
|
||||||
inner: service.into_factory(),
|
inner: service.into_factory(),
|
||||||
}),
|
}),
|
||||||
|
@ -323,8 +322,13 @@ impl ServiceRuntime {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
type BoxServiceFactory =
|
type BoxServiceFactory = service::boxed::BoxServiceFactory<
|
||||||
service::boxed::RcServiceFactory<(), (Option<CounterGuard>, ServerMessage), (), (), ()>;
|
(),
|
||||||
|
(Option<CounterGuard>, ServerMessage),
|
||||||
|
(),
|
||||||
|
(),
|
||||||
|
(),
|
||||||
|
>;
|
||||||
|
|
||||||
struct ServiceFactory<T> {
|
struct ServiceFactory<T> {
|
||||||
inner: T,
|
inner: T,
|
||||||
|
@ -349,7 +353,7 @@ where
|
||||||
let fut = self.inner.create(());
|
let fut = self.inner.create(());
|
||||||
Box::pin(async move {
|
Box::pin(async move {
|
||||||
match fut.await {
|
match fut.await {
|
||||||
Ok(s) => Ok(boxed::rcservice(StreamService::new(s, pool))),
|
Ok(s) => Ok(boxed::service(StreamService::new(s, pool))),
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("Cannot construct service: {:?}", e);
|
error!("Cannot construct service: {:?}", e);
|
||||||
Err(())
|
Err(())
|
||||||
|
|
|
@ -2,10 +2,9 @@ use std::{net::SocketAddr, rc::Rc, task::Context, task::Poll};
|
||||||
|
|
||||||
use log::error;
|
use log::error;
|
||||||
|
|
||||||
use crate::io::Io;
|
use crate::service::{boxed, Ctx, Service, ServiceFactory};
|
||||||
use crate::service::{boxed, Service, ServiceFactory};
|
use crate::util::{BoxFuture, Pool, PoolId};
|
||||||
use crate::util::{BoxFuture, Pool, PoolId, Ready};
|
use crate::{io::Io, time::Millis};
|
||||||
use crate::{rt::spawn, time::Millis};
|
|
||||||
|
|
||||||
use super::{counter::CounterGuard, socket::Stream, Config, Token};
|
use super::{counter::CounterGuard, socket::Stream, Config, Token};
|
||||||
|
|
||||||
|
@ -34,7 +33,7 @@ pub(super) trait InternalServiceFactory: Send {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(super) type BoxedServerService =
|
pub(super) type BoxedServerService =
|
||||||
boxed::RcService<(Option<CounterGuard>, ServerMessage), (), ()>;
|
boxed::BoxService<(Option<CounterGuard>, ServerMessage), (), ()>;
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub(super) struct StreamService<T> {
|
pub(super) struct StreamService<T> {
|
||||||
|
@ -53,12 +52,11 @@ impl<T> StreamService<T> {
|
||||||
|
|
||||||
impl<T> Service<(Option<CounterGuard>, ServerMessage)> for StreamService<T>
|
impl<T> Service<(Option<CounterGuard>, ServerMessage)> for StreamService<T>
|
||||||
where
|
where
|
||||||
T: Service<Io> + 'static,
|
T: Service<Io>,
|
||||||
T::Error: 'static,
|
|
||||||
{
|
{
|
||||||
type Response = ();
|
type Response = ();
|
||||||
type Error = ();
|
type Error = ();
|
||||||
type Future<'f> = Ready<(), ()> where T: 'f;
|
type Future<'f> = BoxFuture<'f, Result<(), ()>> where T: 'f;
|
||||||
|
|
||||||
crate::forward_poll_shutdown!(service);
|
crate::forward_poll_shutdown!(service);
|
||||||
|
|
||||||
|
@ -73,31 +71,31 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(
|
fn call<'a>(
|
||||||
&self,
|
&'a self,
|
||||||
(guard, req): (Option<CounterGuard>, ServerMessage),
|
(guard, req): (Option<CounterGuard>, ServerMessage),
|
||||||
) -> Self::Future<'_> {
|
ctx: Ctx<'a, Self>,
|
||||||
match req {
|
) -> Self::Future<'a> {
|
||||||
ServerMessage::Connect(stream) => {
|
Box::pin(async move {
|
||||||
let stream = stream.try_into().map_err(|e| {
|
match req {
|
||||||
error!("Cannot convert to an async io stream: {}", e);
|
ServerMessage::Connect(stream) => {
|
||||||
});
|
let stream = stream.try_into().map_err(|e| {
|
||||||
|
error!("Cannot convert to an async io stream: {}", e);
|
||||||
if let Ok(stream) = stream {
|
|
||||||
let stream: Io<_> = stream;
|
|
||||||
stream.set_memory_pool(self.pool.pool_ref());
|
|
||||||
let svc = self.service.clone();
|
|
||||||
spawn(async move {
|
|
||||||
let _ = svc.call(stream).await;
|
|
||||||
drop(guard);
|
|
||||||
});
|
});
|
||||||
Ready::Ok(())
|
|
||||||
} else {
|
if let Ok(stream) = stream {
|
||||||
Ready::Err(())
|
let stream: Io<_> = stream;
|
||||||
|
stream.set_memory_pool(self.pool.pool_ref());
|
||||||
|
let _ = ctx.call(self.service.as_ref(), stream).await;
|
||||||
|
drop(guard);
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
_ => Ok(()),
|
||||||
}
|
}
|
||||||
_ => Ready::Ok(()),
|
})
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -153,8 +151,7 @@ where
|
||||||
Box::pin(async move {
|
Box::pin(async move {
|
||||||
match factory.create(()).await {
|
match factory.create(()).await {
|
||||||
Ok(inner) => {
|
Ok(inner) => {
|
||||||
let service: BoxedServerService =
|
let service = boxed::service(StreamService::new(inner, pool));
|
||||||
boxed::rcservice(StreamService::new(inner, pool));
|
|
||||||
Ok(vec![(token, service)])
|
Ok(vec![(token, service)])
|
||||||
}
|
}
|
||||||
Err(_) => Err(()),
|
Err(_) => Err(()),
|
||||||
|
|
|
@ -5,7 +5,7 @@ use async_channel::{unbounded, Receiver, Sender};
|
||||||
use async_oneshot as oneshot;
|
use async_oneshot as oneshot;
|
||||||
|
|
||||||
use crate::rt::{spawn, Arbiter};
|
use crate::rt::{spawn, Arbiter};
|
||||||
use crate::service::Service;
|
use crate::service::Container;
|
||||||
use crate::time::{sleep, Millis, Sleep};
|
use crate::time::{sleep, Millis, Sleep};
|
||||||
use crate::util::{
|
use crate::util::{
|
||||||
join_all, ready, select, stream_recv, BoxFuture, Either, Stream as FutStream,
|
join_all, ready, select, stream_recv, BoxFuture, Either, Stream as FutStream,
|
||||||
|
@ -138,12 +138,12 @@ pub(super) struct Worker {
|
||||||
struct WorkerService {
|
struct WorkerService {
|
||||||
factory: usize,
|
factory: usize,
|
||||||
status: WorkerServiceStatus,
|
status: WorkerServiceStatus,
|
||||||
service: BoxedServerService,
|
service: Container<BoxedServerService>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl WorkerService {
|
impl WorkerService {
|
||||||
fn created(&mut self, service: BoxedServerService) {
|
fn created(&mut self, service: BoxedServerService) {
|
||||||
self.service = service;
|
self.service = Container::new(service);
|
||||||
self.status = WorkerServiceStatus::Unavailable;
|
self.status = WorkerServiceStatus::Unavailable;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -239,7 +239,7 @@ impl Worker {
|
||||||
assert_eq!(token.0, wrk.services.len());
|
assert_eq!(token.0, wrk.services.len());
|
||||||
wrk.services.push(WorkerService {
|
wrk.services.push(WorkerService {
|
||||||
factory,
|
factory,
|
||||||
service,
|
service: service.into(),
|
||||||
status: WorkerServiceStatus::Unavailable,
|
status: WorkerServiceStatus::Unavailable,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -490,9 +490,12 @@ impl Future for Worker {
|
||||||
self.factories[srv.factory].name(msg.token)
|
self.factories[srv.factory].name(msg.token)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
let _ = srv
|
let srv = srv.service.clone();
|
||||||
.service
|
spawn(async move {
|
||||||
.call((Some(guard), ServerMessage::Connect(msg.io)));
|
let _ = srv
|
||||||
|
.call((Some(guard), ServerMessage::Connect(msg.io)))
|
||||||
|
.await;
|
||||||
|
});
|
||||||
} else {
|
} else {
|
||||||
return Poll::Ready(());
|
return Poll::Ready(());
|
||||||
}
|
}
|
||||||
|
@ -509,7 +512,7 @@ mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::io::Io;
|
use crate::io::Io;
|
||||||
use crate::server::service::Factory;
|
use crate::server::service::Factory;
|
||||||
use crate::service::{Service, ServiceFactory};
|
use crate::service::{Ctx, Service, ServiceFactory};
|
||||||
use crate::util::{lazy, Ready};
|
use crate::util::{lazy, Ready};
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug)]
|
#[derive(Clone, Copy, Debug)]
|
||||||
|
@ -569,7 +572,7 @@ mod tests {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, _: Io) -> Self::Future<'_> {
|
fn call<'a>(&'a self, _: Io, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
Ready::Ok(())
|
Ready::Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,7 @@ use crate::http::Request;
|
||||||
use crate::router::ResourceDef;
|
use crate::router::ResourceDef;
|
||||||
use crate::service::boxed::{self, BoxServiceFactory};
|
use crate::service::boxed::{self, BoxServiceFactory};
|
||||||
use crate::service::{map_config, pipeline_factory, IntoServiceFactory, PipelineFactory};
|
use crate::service::{map_config, pipeline_factory, IntoServiceFactory, PipelineFactory};
|
||||||
use crate::service::{Identity, Middleware, Service, ServiceFactory, Stack};
|
use crate::service::{Ctx, Identity, Middleware, Service, ServiceFactory, Stack};
|
||||||
use crate::util::{BoxFuture, Extensions, Ready};
|
use crate::util::{BoxFuture, Extensions, Ready};
|
||||||
|
|
||||||
use super::app_service::{AppFactory, AppService};
|
use super::app_service::{AppFactory, AppService};
|
||||||
|
@ -581,7 +581,7 @@ impl<Err: ErrorRenderer> Service<WebRequest<Err>> for Filter<Err> {
|
||||||
type Future<'f> = Ready<WebRequest<Err>, Err::Container>;
|
type Future<'f> = Ready<WebRequest<Err>, Err::Container>;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: WebRequest<Err>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: WebRequest<Err>, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
Ready::Ok(req)
|
Ready::Ok(req)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -591,7 +591,7 @@ mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::http::header::{self, HeaderValue};
|
use crate::http::header::{self, HeaderValue};
|
||||||
use crate::http::{Method, StatusCode};
|
use crate::http::{Method, StatusCode};
|
||||||
use crate::service::{fn_service, Service};
|
use crate::service::fn_service;
|
||||||
use crate::util::{Bytes, Ready};
|
use crate::util::{Bytes, Ready};
|
||||||
use crate::web::test::{call_service, init_service, read_body, TestRequest};
|
use crate::web::test::{call_service, init_service, read_body, TestRequest};
|
||||||
use crate::web::{
|
use crate::web::{
|
||||||
|
@ -604,7 +604,7 @@ mod tests {
|
||||||
let srv = App::new()
|
let srv = App::new()
|
||||||
.service(web::resource("/test").to(|| async { HttpResponse::Ok() }))
|
.service(web::resource("/test").to(|| async { HttpResponse::Ok() }))
|
||||||
.finish()
|
.finish()
|
||||||
.create(())
|
.container(())
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
let req = TestRequest::with_uri("/test").to_request();
|
let req = TestRequest::with_uri("/test").to_request();
|
||||||
|
@ -628,7 +628,7 @@ mod tests {
|
||||||
Ok(r.into_response(HttpResponse::MethodNotAllowed()))
|
Ok(r.into_response(HttpResponse::MethodNotAllowed()))
|
||||||
})
|
})
|
||||||
.with_config(Default::default())
|
.with_config(Default::default())
|
||||||
.create(())
|
.container(())
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
|
|
|
@ -4,8 +4,10 @@ use std::{cell::RefCell, future::Future, marker::PhantomData, pin::Pin, rc::Rc};
|
||||||
use crate::http::{Request, Response};
|
use crate::http::{Request, Response};
|
||||||
use crate::router::{Path, ResourceDef, Router};
|
use crate::router::{Path, ResourceDef, Router};
|
||||||
use crate::service::boxed::{self, BoxService, BoxServiceFactory};
|
use crate::service::boxed::{self, BoxService, BoxServiceFactory};
|
||||||
use crate::service::{fn_service, Middleware, PipelineFactory, Service, ServiceFactory};
|
use crate::service::{
|
||||||
use crate::util::{BoxFuture, Extensions};
|
fn_service, Ctx, Middleware, PipelineFactory, Service, ServiceCall, ServiceFactory,
|
||||||
|
};
|
||||||
|
use crate::util::{BoxFuture, Either, Extensions};
|
||||||
|
|
||||||
use super::config::AppConfig;
|
use super::config::AppConfig;
|
||||||
use super::error::ErrorRenderer;
|
use super::error::ErrorRenderer;
|
||||||
|
@ -21,8 +23,8 @@ type HttpService<Err: ErrorRenderer> =
|
||||||
BoxService<WebRequest<Err>, WebResponse, Err::Container>;
|
BoxService<WebRequest<Err>, WebResponse, Err::Container>;
|
||||||
type HttpNewService<Err: ErrorRenderer> =
|
type HttpNewService<Err: ErrorRenderer> =
|
||||||
BoxServiceFactory<(), WebRequest<Err>, WebResponse, Err::Container, ()>;
|
BoxServiceFactory<(), WebRequest<Err>, WebResponse, Err::Container, ()>;
|
||||||
type BoxResponse<'f, Err: ErrorRenderer> =
|
type BoxResponse<'a, Err: ErrorRenderer> =
|
||||||
BoxFuture<'f, Result<WebResponse, Err::Container>>;
|
ServiceCall<'a, HttpService<Err>, WebRequest<Err>>;
|
||||||
type FnStateFactory = Box<dyn Fn(Extensions) -> BoxFuture<'static, Result<Extensions, ()>>>;
|
type FnStateFactory = Box<dyn Fn(Extensions) -> BoxFuture<'static, Result<Extensions, ()>>>;
|
||||||
|
|
||||||
/// Service factory to convert `Request` to a `WebRequest<S>`.
|
/// Service factory to convert `Request` to a `WebRequest<S>`.
|
||||||
|
@ -198,12 +200,12 @@ where
|
||||||
{
|
{
|
||||||
type Response = WebResponse;
|
type Response = WebResponse;
|
||||||
type Error = T::Error;
|
type Error = T::Error;
|
||||||
type Future<'f> = T::Future<'f> where T: 'f;
|
type Future<'f> = ServiceCall<'f, T, WebRequest<Err>> where T: 'f;
|
||||||
|
|
||||||
crate::forward_poll_ready!(service);
|
crate::forward_poll_ready!(service);
|
||||||
crate::forward_poll_shutdown!(service);
|
crate::forward_poll_shutdown!(service);
|
||||||
|
|
||||||
fn call(&self, req: Request) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: Request, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
let (head, payload) = req.into_parts();
|
let (head, payload) = req.into_parts();
|
||||||
|
|
||||||
let req = if let Some(mut req) = self.pool.get_request() {
|
let req = if let Some(mut req) = self.pool.get_request() {
|
||||||
|
@ -223,7 +225,7 @@ where
|
||||||
self.pool,
|
self.pool,
|
||||||
)
|
)
|
||||||
};
|
};
|
||||||
self.service.call(WebRequest::new(req))
|
ctx.call(&self.service, WebRequest::new(req))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -245,9 +247,14 @@ struct AppRouting<Err: ErrorRenderer> {
|
||||||
impl<Err: ErrorRenderer> Service<WebRequest<Err>> for AppRouting<Err> {
|
impl<Err: ErrorRenderer> Service<WebRequest<Err>> for AppRouting<Err> {
|
||||||
type Response = WebResponse;
|
type Response = WebResponse;
|
||||||
type Error = Err::Container;
|
type Error = Err::Container;
|
||||||
type Future<'f> = BoxResponse<'f, Err>;
|
type Future<'f> =
|
||||||
|
Either<BoxResponse<'f, Err>, BoxFuture<'f, Result<WebResponse, Err::Container>>>;
|
||||||
|
|
||||||
fn call(&self, mut req: WebRequest<Err>) -> Self::Future<'_> {
|
fn call<'a>(
|
||||||
|
&'a self,
|
||||||
|
mut req: WebRequest<Err>,
|
||||||
|
ctx: Ctx<'a, Self>,
|
||||||
|
) -> Self::Future<'a> {
|
||||||
let res = self.router.recognize_checked(&mut req, |req, guards| {
|
let res = self.router.recognize_checked(&mut req, |req, guards| {
|
||||||
if let Some(guards) = guards {
|
if let Some(guards) = guards {
|
||||||
for f in guards {
|
for f in guards {
|
||||||
|
@ -260,12 +267,14 @@ impl<Err: ErrorRenderer> Service<WebRequest<Err>> for AppRouting<Err> {
|
||||||
});
|
});
|
||||||
|
|
||||||
if let Some((srv, _info)) = res {
|
if let Some((srv, _info)) = res {
|
||||||
srv.call(req)
|
Either::Left(ctx.call(srv, req))
|
||||||
} else if let Some(ref default) = self.default {
|
} else if let Some(ref default) = self.default {
|
||||||
default.call(req)
|
Either::Left(ctx.call(default, req))
|
||||||
} else {
|
} else {
|
||||||
let req = req.into_parts().0;
|
let req = req.into_parts().0;
|
||||||
Box::pin(async { Ok(WebResponse::new(Response::NotFound().finish(), req)) })
|
Either::Right(Box::pin(async {
|
||||||
|
Ok(WebResponse::new(Response::NotFound().finish(), req))
|
||||||
|
}))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -296,23 +305,28 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, req: WebRequest<Err>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: WebRequest<Err>, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
AppServiceResponse {
|
AppServiceResponse {
|
||||||
filter: self.filter.call(req),
|
filter: ctx.call(&self.filter, req),
|
||||||
routing: &self.routing,
|
routing: &self.routing,
|
||||||
endpoint: None,
|
endpoint: None,
|
||||||
|
ctx,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type BoxAppServiceResponse<'a, Err: ErrorRenderer> =
|
||||||
|
ServiceCall<'a, AppRouting<Err>, WebRequest<Err>>;
|
||||||
|
|
||||||
pin_project_lite::pin_project! {
|
pin_project_lite::pin_project! {
|
||||||
pub struct AppServiceResponse<'f, F: Service<WebRequest<Err>>, Err: ErrorRenderer>
|
pub struct AppServiceResponse<'f, F: Service<WebRequest<Err>>, Err: ErrorRenderer>
|
||||||
where F: 'f
|
where F: 'f
|
||||||
{
|
{
|
||||||
#[pin]
|
#[pin]
|
||||||
filter: F::Future<'f>,
|
filter: ServiceCall<'f, F, WebRequest<Err>>,
|
||||||
routing: &'f AppRouting<Err>,
|
routing: &'f AppRouting<Err>,
|
||||||
endpoint: Option<BoxResponse<'f, Err>>,
|
endpoint: Option<BoxAppServiceResponse<'f, Err>>,
|
||||||
|
ctx: Ctx<'f, AppService<F, Err>>,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -335,7 +349,7 @@ where
|
||||||
} else {
|
} else {
|
||||||
return Poll::Pending;
|
return Poll::Pending;
|
||||||
};
|
};
|
||||||
*this.endpoint = Some(this.routing.call(res));
|
*this.endpoint = Some(this.ctx.call(this.routing, res));
|
||||||
this = self.as_mut().project();
|
this = self.as_mut().project();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -347,7 +361,6 @@ mod tests {
|
||||||
use std::sync::atomic::{AtomicBool, Ordering};
|
use std::sync::atomic::{AtomicBool, Ordering};
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
use crate::service::Service;
|
|
||||||
use crate::web::test::{init_service, TestRequest};
|
use crate::web::test::{init_service, TestRequest};
|
||||||
use crate::web::{self, App, HttpResponse};
|
use crate::web::{self, App, HttpResponse};
|
||||||
|
|
||||||
|
|
|
@ -129,9 +129,9 @@ impl<Err: ErrorRenderer> ServiceConfig<Err> {
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::http::{Method, StatusCode};
|
use crate::http::{Method, StatusCode};
|
||||||
|
use crate::util::Bytes;
|
||||||
use crate::web::test::{call_service, init_service, read_body, TestRequest};
|
use crate::web::test::{call_service, init_service, read_body, TestRequest};
|
||||||
use crate::web::{self, App, HttpRequest, HttpResponse};
|
use crate::web::{self, App, HttpRequest, HttpResponse};
|
||||||
use crate::{service::Service, util::Bytes};
|
|
||||||
|
|
||||||
#[crate::rt_test]
|
#[crate::rt_test]
|
||||||
async fn test_configure_state() {
|
async fn test_configure_state() {
|
||||||
|
|
|
@ -4,7 +4,7 @@ use std::{cmp, future::Future, marker, pin::Pin, str::FromStr};
|
||||||
|
|
||||||
use crate::http::encoding::Encoder;
|
use crate::http::encoding::Encoder;
|
||||||
use crate::http::header::{ContentEncoding, ACCEPT_ENCODING};
|
use crate::http::header::{ContentEncoding, ACCEPT_ENCODING};
|
||||||
use crate::service::{Middleware, Service};
|
use crate::service::{Ctx, Middleware, Service, ServiceCall};
|
||||||
use crate::web::{BodyEncoding, ErrorRenderer, WebRequest, WebResponse};
|
use crate::web::{BodyEncoding, ErrorRenderer, WebRequest, WebResponse};
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
|
@ -71,7 +71,7 @@ where
|
||||||
crate::forward_poll_ready!(service);
|
crate::forward_poll_ready!(service);
|
||||||
crate::forward_poll_shutdown!(service);
|
crate::forward_poll_shutdown!(service);
|
||||||
|
|
||||||
fn call(&self, req: WebRequest<E>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: WebRequest<E>, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
// negotiate content-encoding
|
// negotiate content-encoding
|
||||||
let encoding = if let Some(val) = req.headers().get(&ACCEPT_ENCODING) {
|
let encoding = if let Some(val) = req.headers().get(&ACCEPT_ENCODING) {
|
||||||
if let Ok(enc) = val.to_str() {
|
if let Ok(enc) = val.to_str() {
|
||||||
|
@ -85,7 +85,7 @@ where
|
||||||
|
|
||||||
CompressResponse {
|
CompressResponse {
|
||||||
encoding,
|
encoding,
|
||||||
fut: self.service.call(req),
|
fut: ctx.call(&self.service, req),
|
||||||
_t: marker::PhantomData,
|
_t: marker::PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -97,7 +97,7 @@ pin_project_lite::pin_project! {
|
||||||
where S: 'f, E: 'f
|
where S: 'f, E: 'f
|
||||||
{
|
{
|
||||||
#[pin]
|
#[pin]
|
||||||
fut: S::Future<'f>,
|
fut: ServiceCall<'f, S, WebRequest<E>>,
|
||||||
encoding: ContentEncoding,
|
encoding: ContentEncoding,
|
||||||
_t: marker::PhantomData<E>,
|
_t: marker::PhantomData<E>,
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,7 +3,7 @@ use std::rc::Rc;
|
||||||
|
|
||||||
use crate::http::error::HttpError;
|
use crate::http::error::HttpError;
|
||||||
use crate::http::header::{HeaderMap, HeaderName, HeaderValue, CONTENT_TYPE};
|
use crate::http::header::{HeaderMap, HeaderName, HeaderValue, CONTENT_TYPE};
|
||||||
use crate::service::{Middleware, Service};
|
use crate::service::{Ctx, Middleware, Service};
|
||||||
use crate::util::BoxFuture;
|
use crate::util::BoxFuture;
|
||||||
use crate::web::{WebRequest, WebResponse};
|
use crate::web::{WebRequest, WebResponse};
|
||||||
|
|
||||||
|
@ -115,9 +115,9 @@ where
|
||||||
crate::forward_poll_ready!(service);
|
crate::forward_poll_ready!(service);
|
||||||
crate::forward_poll_shutdown!(service);
|
crate::forward_poll_shutdown!(service);
|
||||||
|
|
||||||
fn call(&self, req: WebRequest<E>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: WebRequest<E>, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
Box::pin(async move {
|
Box::pin(async move {
|
||||||
let mut res = self.service.call(req).await?;
|
let mut res = ctx.call(&self.service, req).await?;
|
||||||
|
|
||||||
// set response headers
|
// set response headers
|
||||||
for (key, value) in self.inner.headers.iter() {
|
for (key, value) in self.inner.headers.iter() {
|
||||||
|
@ -141,7 +141,7 @@ where
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::http::header::CONTENT_TYPE;
|
use crate::http::header::CONTENT_TYPE;
|
||||||
use crate::service::IntoService;
|
use crate::service::{Container, IntoService};
|
||||||
use crate::util::lazy;
|
use crate::util::lazy;
|
||||||
use crate::web::request::WebRequest;
|
use crate::web::request::WebRequest;
|
||||||
use crate::web::test::{ok_service, TestRequest};
|
use crate::web::test::{ok_service, TestRequest};
|
||||||
|
@ -149,9 +149,11 @@ mod tests {
|
||||||
|
|
||||||
#[crate::rt_test]
|
#[crate::rt_test]
|
||||||
async fn test_default_headers() {
|
async fn test_default_headers() {
|
||||||
let mw = DefaultHeaders::new()
|
let mw = Container::new(
|
||||||
.header(CONTENT_TYPE, "0001")
|
DefaultHeaders::new()
|
||||||
.create(ok_service());
|
.header(CONTENT_TYPE, "0001")
|
||||||
|
.create(ok_service()),
|
||||||
|
);
|
||||||
|
|
||||||
assert!(lazy(|cx| mw.poll_ready(cx).is_ready()).await);
|
assert!(lazy(|cx| mw.poll_ready(cx).is_ready()).await);
|
||||||
assert!(lazy(|cx| mw.poll_shutdown(cx).is_ready()).await);
|
assert!(lazy(|cx| mw.poll_shutdown(cx).is_ready()).await);
|
||||||
|
@ -166,9 +168,11 @@ mod tests {
|
||||||
req.into_response(HttpResponse::Ok().header(CONTENT_TYPE, "0002").finish()),
|
req.into_response(HttpResponse::Ok().header(CONTENT_TYPE, "0002").finish()),
|
||||||
)
|
)
|
||||||
};
|
};
|
||||||
let mw = DefaultHeaders::new()
|
let mw = Container::new(
|
||||||
.header(CONTENT_TYPE, "0001")
|
DefaultHeaders::new()
|
||||||
.create(srv.into_service());
|
.header(CONTENT_TYPE, "0001")
|
||||||
|
.create(srv.into_service()),
|
||||||
|
);
|
||||||
let resp = mw.call(req).await.unwrap();
|
let resp = mw.call(req).await.unwrap();
|
||||||
assert_eq!(resp.headers().get(CONTENT_TYPE).unwrap(), "0002");
|
assert_eq!(resp.headers().get(CONTENT_TYPE).unwrap(), "0002");
|
||||||
}
|
}
|
||||||
|
@ -178,9 +182,11 @@ mod tests {
|
||||||
let srv = |req: WebRequest<DefaultError>| async move {
|
let srv = |req: WebRequest<DefaultError>| async move {
|
||||||
Ok::<_, Error>(req.into_response(HttpResponse::Ok().finish()))
|
Ok::<_, Error>(req.into_response(HttpResponse::Ok().finish()))
|
||||||
};
|
};
|
||||||
let mw = DefaultHeaders::new()
|
let mw = Container::new(
|
||||||
.content_type()
|
DefaultHeaders::new()
|
||||||
.create(srv.into_service());
|
.content_type()
|
||||||
|
.create(srv.into_service()),
|
||||||
|
);
|
||||||
|
|
||||||
let req = TestRequest::default().to_srv_request();
|
let req = TestRequest::default().to_srv_request();
|
||||||
let resp = mw.call(req).await.unwrap();
|
let resp = mw.call(req).await.unwrap();
|
||||||
|
|
|
@ -7,7 +7,7 @@ use regex::Regex;
|
||||||
|
|
||||||
use crate::http::body::{Body, BodySize, MessageBody, ResponseBody};
|
use crate::http::body::{Body, BodySize, MessageBody, ResponseBody};
|
||||||
use crate::http::header::HeaderName;
|
use crate::http::header::HeaderName;
|
||||||
use crate::service::{Middleware, Service};
|
use crate::service::{Ctx, Middleware, Service, ServiceCall};
|
||||||
use crate::util::{Bytes, Either, HashSet};
|
use crate::util::{Bytes, Either, HashSet};
|
||||||
use crate::web::{HttpResponse, WebRequest, WebResponse};
|
use crate::web::{HttpResponse, WebRequest, WebResponse};
|
||||||
|
|
||||||
|
@ -136,15 +136,15 @@ where
|
||||||
{
|
{
|
||||||
type Response = WebResponse;
|
type Response = WebResponse;
|
||||||
type Error = S::Error;
|
type Error = S::Error;
|
||||||
type Future<'f> = Either<LoggerResponse<'f, S, E>, S::Future<'f>> where S: 'f, E: 'f;
|
type Future<'f> = Either<LoggerResponse<'f, S, E>, ServiceCall<'f, S, WebRequest<E>>> where S: 'f, E: 'f;
|
||||||
|
|
||||||
crate::forward_poll_ready!(service);
|
crate::forward_poll_ready!(service);
|
||||||
crate::forward_poll_shutdown!(service);
|
crate::forward_poll_shutdown!(service);
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: WebRequest<E>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: WebRequest<E>, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
if self.inner.exclude.contains(req.path()) {
|
if self.inner.exclude.contains(req.path()) {
|
||||||
Either::Right(self.service.call(req))
|
Either::Right(ctx.call(&self.service, req))
|
||||||
} else {
|
} else {
|
||||||
let time = time::SystemTime::now();
|
let time = time::SystemTime::now();
|
||||||
let mut format = self.inner.format.clone();
|
let mut format = self.inner.format.clone();
|
||||||
|
@ -155,7 +155,7 @@ where
|
||||||
Either::Left(LoggerResponse {
|
Either::Left(LoggerResponse {
|
||||||
time,
|
time,
|
||||||
format: Some(format),
|
format: Some(format),
|
||||||
fut: self.service.call(req),
|
fut: ctx.call(&self.service, req),
|
||||||
_t: PhantomData,
|
_t: PhantomData,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -168,7 +168,7 @@ pin_project_lite::pin_project! {
|
||||||
where S: 'f, E: 'f
|
where S: 'f, E: 'f
|
||||||
{
|
{
|
||||||
#[pin]
|
#[pin]
|
||||||
fut: S::Future<'f>,
|
fut: ServiceCall<'f, S, WebRequest<E>>,
|
||||||
time: time::SystemTime,
|
time: time::SystemTime,
|
||||||
format: Option<Format>,
|
format: Option<Format>,
|
||||||
_t: PhantomData<E>
|
_t: PhantomData<E>
|
||||||
|
@ -448,7 +448,7 @@ impl<'a> fmt::Display for FormatDisplay<'a> {
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::http::{header, StatusCode};
|
use crate::http::{header, StatusCode};
|
||||||
use crate::service::{IntoService, Middleware};
|
use crate::service::{Container, IntoService, Middleware};
|
||||||
use crate::util::lazy;
|
use crate::util::lazy;
|
||||||
use crate::web::test::{self, TestRequest};
|
use crate::web::test::{self, TestRequest};
|
||||||
use crate::web::{DefaultError, Error};
|
use crate::web::{DefaultError, Error};
|
||||||
|
@ -468,7 +468,7 @@ mod tests {
|
||||||
let logger = Logger::new("%% %{User-Agent}i %{X-Test}o %{HOME}e %D %% test")
|
let logger = Logger::new("%% %{User-Agent}i %{X-Test}o %{HOME}e %D %% test")
|
||||||
.exclude("/test");
|
.exclude("/test");
|
||||||
|
|
||||||
let srv = Middleware::create(&logger, srv.into_service());
|
let srv = Container::new(Middleware::create(&logger, srv.into_service()));
|
||||||
assert!(lazy(|cx| srv.poll_ready(cx).is_ready()).await);
|
assert!(lazy(|cx| srv.poll_ready(cx).is_ready()).await);
|
||||||
assert!(lazy(|cx| srv.poll_shutdown(cx).is_ready()).await);
|
assert!(lazy(|cx| srv.poll_shutdown(cx).is_ready()).await);
|
||||||
|
|
||||||
|
|
|
@ -3,9 +3,11 @@ use std::{cell::RefCell, fmt, rc::Rc};
|
||||||
use crate::http::Response;
|
use crate::http::Response;
|
||||||
use crate::router::{IntoPattern, ResourceDef};
|
use crate::router::{IntoPattern, ResourceDef};
|
||||||
use crate::service::boxed::{self, BoxService, BoxServiceFactory};
|
use crate::service::boxed::{self, BoxService, BoxServiceFactory};
|
||||||
use crate::service::{dev::AndThen, pipeline, pipeline_factory, Pipeline, PipelineFactory};
|
|
||||||
use crate::service::{
|
use crate::service::{
|
||||||
Identity, IntoServiceFactory, Middleware, Service, ServiceFactory, Stack,
|
dev::AndThen, pipeline, pipeline_factory, Ctx, Pipeline, PipelineFactory,
|
||||||
|
};
|
||||||
|
use crate::service::{
|
||||||
|
Identity, IntoServiceFactory, Middleware, Service, ServiceCall, ServiceFactory, Stack,
|
||||||
};
|
};
|
||||||
use crate::util::{BoxFuture, Either, Extensions, Ready};
|
use crate::util::{BoxFuture, Either, Extensions, Ready};
|
||||||
|
|
||||||
|
@ -22,6 +24,8 @@ type HttpService<Err: ErrorRenderer> =
|
||||||
type HttpNewService<Err: ErrorRenderer> =
|
type HttpNewService<Err: ErrorRenderer> =
|
||||||
BoxServiceFactory<(), WebRequest<Err>, WebResponse, Err::Container, ()>;
|
BoxServiceFactory<(), WebRequest<Err>, WebResponse, Err::Container, ()>;
|
||||||
type ResourcePipeline<F, Err> = Pipeline<WebRequest<Err>, AndThen<F, ResourceRouter<Err>>>;
|
type ResourcePipeline<F, Err> = Pipeline<WebRequest<Err>, AndThen<F, ResourceRouter<Err>>>;
|
||||||
|
type BoxResponse<'a, Err: ErrorRenderer> =
|
||||||
|
ServiceCall<'a, HttpService<Err>, WebRequest<Err>>;
|
||||||
|
|
||||||
/// *Resource* is an entry in resources table which corresponds to requested URL.
|
/// *Resource* is an entry in resources table which corresponds to requested URL.
|
||||||
///
|
///
|
||||||
|
@ -456,6 +460,9 @@ impl<Err: ErrorRenderer> ServiceFactory<WebRequest<Err>> for ResourceRouterFacto
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type BoxResourceRouterResponse<'a, Err: ErrorRenderer> =
|
||||||
|
ServiceCall<'a, RouteService<Err>, WebRequest<Err>>;
|
||||||
|
|
||||||
pub struct ResourceRouter<Err: ErrorRenderer> {
|
pub struct ResourceRouter<Err: ErrorRenderer> {
|
||||||
state: Option<AppState>,
|
state: Option<AppState>,
|
||||||
routes: Vec<RouteService<Err>>,
|
routes: Vec<RouteService<Err>>,
|
||||||
|
@ -466,26 +473,30 @@ impl<Err: ErrorRenderer> Service<WebRequest<Err>> for ResourceRouter<Err> {
|
||||||
type Response = WebResponse;
|
type Response = WebResponse;
|
||||||
type Error = Err::Container;
|
type Error = Err::Container;
|
||||||
type Future<'f> = Either<
|
type Future<'f> = Either<
|
||||||
Ready<WebResponse, Err::Container>,
|
BoxResourceRouterResponse<'f, Err>,
|
||||||
BoxFuture<'f, Result<WebResponse, Err::Container>>,
|
Either<Ready<WebResponse, Err::Container>, BoxResponse<'f, Err>>,
|
||||||
>;
|
>;
|
||||||
|
|
||||||
fn call(&self, mut req: WebRequest<Err>) -> Self::Future<'_> {
|
fn call<'a>(
|
||||||
|
&'a self,
|
||||||
|
mut req: WebRequest<Err>,
|
||||||
|
ctx: Ctx<'a, Self>,
|
||||||
|
) -> Self::Future<'a> {
|
||||||
for route in self.routes.iter() {
|
for route in self.routes.iter() {
|
||||||
if route.check(&mut req) {
|
if route.check(&mut req) {
|
||||||
if let Some(ref state) = self.state {
|
if let Some(ref state) = self.state {
|
||||||
req.set_state_container(state.clone());
|
req.set_state_container(state.clone());
|
||||||
}
|
}
|
||||||
return Either::Right(route.call(req));
|
return Either::Left(ctx.call(route, req));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if let Some(ref default) = self.default {
|
if let Some(ref default) = self.default {
|
||||||
Either::Right(default.call(req))
|
Either::Right(Either::Right(ctx.call(default, req)))
|
||||||
} else {
|
} else {
|
||||||
Either::Left(Ready::Ok(WebResponse::new(
|
Either::Right(Either::Left(Ready::Ok(WebResponse::new(
|
||||||
Response::MethodNotAllowed().finish(),
|
Response::MethodNotAllowed().finish(),
|
||||||
req.into_parts().0,
|
req.into_parts().0,
|
||||||
)))
|
))))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -404,7 +404,7 @@ pub(crate) mod tests {
|
||||||
use crate::http::header::{HeaderValue, CONTENT_TYPE};
|
use crate::http::header::{HeaderValue, CONTENT_TYPE};
|
||||||
use crate::http::{Response as HttpResponse, StatusCode};
|
use crate::http::{Response as HttpResponse, StatusCode};
|
||||||
use crate::web::test::{init_service, TestRequest};
|
use crate::web::test::{init_service, TestRequest};
|
||||||
use crate::{service::Service, util::Bytes, util::BytesMut, web};
|
use crate::{util::Bytes, util::BytesMut, web};
|
||||||
|
|
||||||
fn responder<T: Responder<DefaultError>>(responder: T) -> impl Responder<DefaultError> {
|
fn responder<T: Responder<DefaultError>>(responder: T) -> impl Responder<DefaultError> {
|
||||||
responder
|
responder
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use std::{mem, rc::Rc};
|
use std::{mem, rc::Rc};
|
||||||
|
|
||||||
use crate::util::{BoxFuture, Ready};
|
use crate::util::{BoxFuture, Ready};
|
||||||
use crate::{http::Method, service::Service, service::ServiceFactory};
|
use crate::{http::Method, service::Ctx, service::Service, service::ServiceFactory};
|
||||||
|
|
||||||
use super::error::ErrorRenderer;
|
use super::error::ErrorRenderer;
|
||||||
use super::error_default::DefaultError;
|
use super::error_default::DefaultError;
|
||||||
|
@ -90,7 +90,7 @@ impl<Err: ErrorRenderer> Service<WebRequest<Err>> for RouteService<Err> {
|
||||||
type Future<'f> = BoxFuture<'f, Result<Self::Response, Self::Error>>;
|
type Future<'f> = BoxFuture<'f, Result<Self::Response, Self::Error>>;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn call(&self, req: WebRequest<Err>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: WebRequest<Err>, _: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
self.handler.call(req)
|
self.handler.call(req)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,9 @@ use crate::http::Response;
|
||||||
use crate::router::{IntoPattern, ResourceDef, Router};
|
use crate::router::{IntoPattern, ResourceDef, Router};
|
||||||
use crate::service::boxed::{self, BoxService, BoxServiceFactory};
|
use crate::service::boxed::{self, BoxService, BoxServiceFactory};
|
||||||
use crate::service::{pipeline_factory, IntoServiceFactory, PipelineFactory};
|
use crate::service::{pipeline_factory, IntoServiceFactory, PipelineFactory};
|
||||||
use crate::service::{Identity, Middleware, Service, ServiceFactory, Stack};
|
use crate::service::{
|
||||||
|
Ctx, Identity, Middleware, Service, ServiceCall, ServiceFactory, Stack,
|
||||||
|
};
|
||||||
use crate::util::{BoxFuture, Either, Extensions, Ready};
|
use crate::util::{BoxFuture, Either, Extensions, Ready};
|
||||||
|
|
||||||
use super::app::Filter;
|
use super::app::Filter;
|
||||||
|
@ -27,7 +29,7 @@ type HttpService<Err: ErrorRenderer> =
|
||||||
type HttpNewService<Err: ErrorRenderer> =
|
type HttpNewService<Err: ErrorRenderer> =
|
||||||
BoxServiceFactory<(), WebRequest<Err>, WebResponse, Err::Container, ()>;
|
BoxServiceFactory<(), WebRequest<Err>, WebResponse, Err::Container, ()>;
|
||||||
type BoxResponse<'a, Err: ErrorRenderer> =
|
type BoxResponse<'a, Err: ErrorRenderer> =
|
||||||
BoxFuture<'a, Result<WebResponse, Err::Container>>;
|
ServiceCall<'a, HttpService<Err>, WebRequest<Err>>;
|
||||||
|
|
||||||
/// Resources scope.
|
/// Resources scope.
|
||||||
///
|
///
|
||||||
|
@ -503,11 +505,12 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, req: WebRequest<Err>) -> Self::Future<'_> {
|
fn call<'a>(&'a self, req: WebRequest<Err>, ctx: Ctx<'a, Self>) -> Self::Future<'a> {
|
||||||
ScopeServiceResponse {
|
ScopeServiceResponse {
|
||||||
filter: self.filter.call(req),
|
filter: ctx.call(&self.filter, req),
|
||||||
routing: &self.routing,
|
routing: &self.routing,
|
||||||
endpoint: None,
|
endpoint: None,
|
||||||
|
ctx,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -517,9 +520,10 @@ pin_project_lite::pin_project! {
|
||||||
where F: 'f
|
where F: 'f
|
||||||
{
|
{
|
||||||
#[pin]
|
#[pin]
|
||||||
filter: F::Future<'f>,
|
filter: ServiceCall<'f, F, WebRequest<Err>>,
|
||||||
routing: &'f ScopeRouter<Err>,
|
routing: &'f ScopeRouter<Err>,
|
||||||
endpoint: Option<<ScopeRouter<Err> as Service<WebRequest<Err>>>::Future<'f>>,
|
ctx: Ctx<'f, ScopeService<F, Err>>,
|
||||||
|
endpoint: Option<ServiceCall<'f, ScopeRouter<Err>, WebRequest<Err>>>,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -536,15 +540,15 @@ where
|
||||||
loop {
|
loop {
|
||||||
if let Some(fut) = this.endpoint.as_mut() {
|
if let Some(fut) = this.endpoint.as_mut() {
|
||||||
return Pin::new(fut).poll(cx);
|
return Pin::new(fut).poll(cx);
|
||||||
} else {
|
|
||||||
let res = if let Poll::Ready(res) = this.filter.poll(cx) {
|
|
||||||
res?
|
|
||||||
} else {
|
|
||||||
return Poll::Pending;
|
|
||||||
};
|
|
||||||
*this.endpoint = Some(this.routing.call(res));
|
|
||||||
this = self.as_mut().project();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let res = if let Poll::Ready(res) = this.filter.poll(cx) {
|
||||||
|
res?
|
||||||
|
} else {
|
||||||
|
return Poll::Pending;
|
||||||
|
};
|
||||||
|
*this.endpoint = Some(this.ctx.call(this.routing, res));
|
||||||
|
this = self.as_mut().project();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -601,7 +605,11 @@ impl<Err: ErrorRenderer> Service<WebRequest<Err>> for ScopeRouter<Err> {
|
||||||
type Error = Err::Container;
|
type Error = Err::Container;
|
||||||
type Future<'f> = Either<BoxResponse<'f, Err>, Ready<Self::Response, Self::Error>>;
|
type Future<'f> = Either<BoxResponse<'f, Err>, Ready<Self::Response, Self::Error>>;
|
||||||
|
|
||||||
fn call(&self, mut req: WebRequest<Err>) -> Self::Future<'_> {
|
fn call<'a>(
|
||||||
|
&'a self,
|
||||||
|
mut req: WebRequest<Err>,
|
||||||
|
ctx: Ctx<'a, Self>,
|
||||||
|
) -> Self::Future<'a> {
|
||||||
let res = self.router.recognize_checked(&mut req, |req, guards| {
|
let res = self.router.recognize_checked(&mut req, |req, guards| {
|
||||||
if let Some(guards) = guards {
|
if let Some(guards) = guards {
|
||||||
for f in guards {
|
for f in guards {
|
||||||
|
@ -617,9 +625,9 @@ impl<Err: ErrorRenderer> Service<WebRequest<Err>> for ScopeRouter<Err> {
|
||||||
if let Some(ref state) = self.state {
|
if let Some(ref state) = self.state {
|
||||||
req.set_state_container(state.clone());
|
req.set_state_container(state.clone());
|
||||||
}
|
}
|
||||||
Either::Left(srv.call(req))
|
Either::Left(ctx.call(srv, req))
|
||||||
} else if let Some(ref default) = self.default {
|
} else if let Some(ref default) = self.default {
|
||||||
Either::Left(default.call(req))
|
Either::Left(ctx.call(default, req))
|
||||||
} else {
|
} else {
|
||||||
let req = req.into_parts().0;
|
let req = req.into_parts().0;
|
||||||
Either::Right(Ready::Ok(WebResponse::new(
|
Either::Right(Ready::Ok(WebResponse::new(
|
||||||
|
@ -635,7 +643,7 @@ mod tests {
|
||||||
use crate::http::body::{Body, ResponseBody};
|
use crate::http::body::{Body, ResponseBody};
|
||||||
use crate::http::header::{HeaderValue, CONTENT_TYPE};
|
use crate::http::header::{HeaderValue, CONTENT_TYPE};
|
||||||
use crate::http::{Method, StatusCode};
|
use crate::http::{Method, StatusCode};
|
||||||
use crate::service::{fn_service, Service};
|
use crate::service::fn_service;
|
||||||
use crate::util::{Bytes, Ready};
|
use crate::util::{Bytes, Ready};
|
||||||
use crate::web::middleware::DefaultHeaders;
|
use crate::web::middleware::DefaultHeaders;
|
||||||
use crate::web::request::WebRequest;
|
use crate::web::request::WebRequest;
|
||||||
|
|
|
@ -392,7 +392,6 @@ tuple_web_service!((0,A),(1,B),(2,C),(3,D),(4,E),(5,F),(6,G),(7,H),(8,I),(9,J),(
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::http::{Method, StatusCode};
|
use crate::http::{Method, StatusCode};
|
||||||
use crate::service::Service;
|
|
||||||
use crate::web::test::{init_service, TestRequest};
|
use crate::web::test::{init_service, TestRequest};
|
||||||
use crate::web::{self, guard, App, DefaultError, HttpResponse};
|
use crate::web::{self, guard, App, DefaultError, HttpResponse};
|
||||||
|
|
||||||
|
|
|
@ -3,8 +3,7 @@ use std::{fmt, net, net::SocketAddr, rc::Rc, sync::mpsc, thread};
|
||||||
|
|
||||||
#[cfg(feature = "cookie")]
|
#[cfg(feature = "cookie")]
|
||||||
use coo_kie::Cookie;
|
use coo_kie::Cookie;
|
||||||
use serde::de::DeserializeOwned;
|
use serde::{de::DeserializeOwned, Serialize};
|
||||||
use serde::Serialize;
|
|
||||||
|
|
||||||
use crate::http::body::MessageBody;
|
use crate::http::body::MessageBody;
|
||||||
use crate::http::client::{Client, ClientRequest, ClientResponse, Connector};
|
use crate::http::client::{Client, ClientRequest, ClientResponse, Connector};
|
||||||
|
@ -14,7 +13,7 @@ use crate::http::test::TestRequest as HttpTestRequest;
|
||||||
use crate::http::{HttpService, Method, Payload, Request, StatusCode, Uri, Version};
|
use crate::http::{HttpService, Method, Payload, Request, StatusCode, Uri, Version};
|
||||||
use crate::router::{Path, ResourceDef};
|
use crate::router::{Path, ResourceDef};
|
||||||
use crate::service::{
|
use crate::service::{
|
||||||
map_config, IntoService, IntoServiceFactory, Service, ServiceFactory,
|
map_config, Container, IntoService, IntoServiceFactory, Service, ServiceFactory,
|
||||||
};
|
};
|
||||||
use crate::time::{sleep, Millis, Seconds};
|
use crate::time::{sleep, Millis, Seconds};
|
||||||
use crate::util::{stream_recv, Bytes, BytesMut, Extensions, Ready, Stream};
|
use crate::util::{stream_recv, Bytes, BytesMut, Extensions, Ready, Stream};
|
||||||
|
@ -70,14 +69,14 @@ pub fn default_service<Err: ErrorRenderer>(
|
||||||
/// ```
|
/// ```
|
||||||
pub async fn init_service<R, S, E>(
|
pub async fn init_service<R, S, E>(
|
||||||
app: R,
|
app: R,
|
||||||
) -> impl Service<Request, Response = WebResponse, Error = E>
|
) -> Container<impl Service<Request, Response = WebResponse, Error = E>>
|
||||||
where
|
where
|
||||||
R: IntoServiceFactory<S, Request, AppConfig>,
|
R: IntoServiceFactory<S, Request, AppConfig>,
|
||||||
S: ServiceFactory<Request, AppConfig, Response = WebResponse, Error = E>,
|
S: ServiceFactory<Request, AppConfig, Response = WebResponse, Error = E>,
|
||||||
S::InitError: std::fmt::Debug,
|
S::InitError: std::fmt::Debug,
|
||||||
{
|
{
|
||||||
let srv = app.into_factory();
|
let srv = app.into_factory();
|
||||||
srv.create(AppConfig::default()).await.unwrap()
|
srv.container(AppConfig::default()).await.unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Calls service and waits for response future completion.
|
/// Calls service and waits for response future completion.
|
||||||
|
@ -103,7 +102,7 @@ where
|
||||||
/// assert_eq!(resp.status(), StatusCode::OK);
|
/// assert_eq!(resp.status(), StatusCode::OK);
|
||||||
/// }
|
/// }
|
||||||
/// ```
|
/// ```
|
||||||
pub async fn call_service<S, R, E>(app: &S, req: R) -> S::Response
|
pub async fn call_service<S, R, E>(app: &Container<S>, req: R) -> S::Response
|
||||||
where
|
where
|
||||||
S: Service<R, Response = WebResponse, Error = E>,
|
S: Service<R, Response = WebResponse, Error = E>,
|
||||||
E: std::fmt::Debug,
|
E: std::fmt::Debug,
|
||||||
|
@ -136,7 +135,7 @@ where
|
||||||
/// assert_eq!(result, Bytes::from_static(b"welcome!"));
|
/// assert_eq!(result, Bytes::from_static(b"welcome!"));
|
||||||
/// }
|
/// }
|
||||||
/// ```
|
/// ```
|
||||||
pub async fn read_response<S>(app: &S, req: Request) -> Bytes
|
pub async fn read_response<S>(app: &Container<S>, req: Request) -> Bytes
|
||||||
where
|
where
|
||||||
S: Service<Request, Response = WebResponse>,
|
S: Service<Request, Response = WebResponse>,
|
||||||
{
|
{
|
||||||
|
@ -235,7 +234,7 @@ where
|
||||||
/// let result: Person = test::read_response_json(&mut app, req).await;
|
/// let result: Person = test::read_response_json(&mut app, req).await;
|
||||||
/// }
|
/// }
|
||||||
/// ```
|
/// ```
|
||||||
pub async fn read_response_json<S, T>(app: &S, req: Request) -> T
|
pub async fn read_response_json<S, T>(app: &Container<S>, req: Request) -> T
|
||||||
where
|
where
|
||||||
S: Service<Request, Response = WebResponse>,
|
S: Service<Request, Response = WebResponse>,
|
||||||
T: DeserializeOwned,
|
T: DeserializeOwned,
|
||||||
|
|
|
@ -159,8 +159,7 @@ impl<Err: ErrorRenderer> FromRequest<Err> for Bytes {
|
||||||
}
|
}
|
||||||
|
|
||||||
let limit = cfg.limit;
|
let limit = cfg.limit;
|
||||||
let fut = HttpMessageBody::new(req, payload).limit(limit);
|
Either::Left(Box::pin(HttpMessageBody::new(req, payload).limit(limit)))
|
||||||
Either::Left(Box::pin(async move { fut.await }))
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -101,7 +101,6 @@ mod tests {
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::http::StatusCode;
|
use crate::http::StatusCode;
|
||||||
use crate::service::Service;
|
|
||||||
use crate::web::test::{self, init_service, TestRequest};
|
use crate::web::test::{self, init_service, TestRequest};
|
||||||
use crate::web::{self, App, HttpResponse};
|
use crate::web::{self, App, HttpResponse};
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ pub use crate::ws::{CloseCode, CloseReason, Frame, Message, WsSink};
|
||||||
|
|
||||||
use crate::http::{body::BodySize, h1, StatusCode};
|
use crate::http::{body::BodySize, h1, StatusCode};
|
||||||
use crate::service::{
|
use crate::service::{
|
||||||
apply_fn, fn_factory_with_config, IntoServiceFactory, Service, ServiceFactory,
|
apply_fn, fn_factory_with_config, IntoServiceFactory, ServiceFactory,
|
||||||
};
|
};
|
||||||
use crate::web::{HttpRequest, HttpResponse};
|
use crate::web::{HttpRequest, HttpResponse};
|
||||||
use crate::ws::{self, error::HandshakeError, error::WsError, handshake};
|
use crate::ws::{self, error::HandshakeError, error::WsError, handshake};
|
||||||
|
|
|
@ -16,7 +16,7 @@ use crate::http::header::{self, HeaderMap, HeaderName, HeaderValue, AUTHORIZATIO
|
||||||
use crate::http::{body::BodySize, client::ClientResponse, error::HttpError, h1};
|
use crate::http::{body::BodySize, client::ClientResponse, error::HttpError, h1};
|
||||||
use crate::http::{ConnectionType, RequestHead, RequestHeadType, StatusCode, Uri};
|
use crate::http::{ConnectionType, RequestHead, RequestHeadType, StatusCode, Uri};
|
||||||
use crate::io::{Base, DispatchItem, Dispatcher, Filter, Io, Layer, Sealed};
|
use crate::io::{Base, DispatchItem, Dispatcher, Filter, Io, Layer, Sealed};
|
||||||
use crate::service::{apply_fn, into_service, IntoService, Service};
|
use crate::service::{apply_fn, into_service, Container, IntoService, Service};
|
||||||
use crate::time::{timeout, Millis, Seconds};
|
use crate::time::{timeout, Millis, Seconds};
|
||||||
use crate::{channel::mpsc, rt, util::Ready, ws};
|
use crate::{channel::mpsc, rt, util::Ready, ws};
|
||||||
|
|
||||||
|
@ -25,7 +25,7 @@ use super::transport::WsTransport;
|
||||||
|
|
||||||
/// `WebSocket` client builder
|
/// `WebSocket` client builder
|
||||||
pub struct WsClient<F, T> {
|
pub struct WsClient<F, T> {
|
||||||
connector: T,
|
connector: Container<T>,
|
||||||
head: Rc<RequestHead>,
|
head: Rc<RequestHead>,
|
||||||
addr: Option<net::SocketAddr>,
|
addr: Option<net::SocketAddr>,
|
||||||
max_size: usize,
|
max_size: usize,
|
||||||
|
@ -630,7 +630,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(WsClient {
|
Ok(WsClient {
|
||||||
connector: inner.connector,
|
connector: inner.connector.into(),
|
||||||
head: Rc::new(inner.head),
|
head: Rc::new(inner.head),
|
||||||
addr: inner.addr,
|
addr: inner.addr,
|
||||||
max_size: inner.max_size,
|
max_size: inner.max_size,
|
||||||
|
|
|
@ -3,7 +3,7 @@ use std::{io, rc::Rc, sync::Arc};
|
||||||
use ntex::codec::BytesCodec;
|
use ntex::codec::BytesCodec;
|
||||||
use ntex::connect::Connect;
|
use ntex::connect::Connect;
|
||||||
use ntex::io::{types::PeerAddr, Io};
|
use ntex::io::{types::PeerAddr, Io};
|
||||||
use ntex::service::{fn_service, pipeline_factory, Service, ServiceFactory};
|
use ntex::service::{fn_service, pipeline_factory, Container, ServiceFactory};
|
||||||
use ntex::{server::test_server, time, util::Bytes};
|
use ntex::{server::test_server, time, util::Bytes};
|
||||||
|
|
||||||
#[cfg(feature = "openssl")]
|
#[cfg(feature = "openssl")]
|
||||||
|
@ -97,7 +97,7 @@ async fn test_openssl_string() {
|
||||||
let mut builder = SslConnector::builder(SslMethod::tls()).unwrap();
|
let mut builder = SslConnector::builder(SslMethod::tls()).unwrap();
|
||||||
builder.set_verify(SslVerifyMode::NONE);
|
builder.set_verify(SslVerifyMode::NONE);
|
||||||
|
|
||||||
let conn = ntex::connect::openssl::Connector::new(builder.build());
|
let conn = Container::new(ntex::connect::openssl::Connector::new(builder.build()));
|
||||||
let addr = format!("127.0.0.1:{}", srv.addr().port());
|
let addr = format!("127.0.0.1:{}", srv.addr().port());
|
||||||
let io = conn.call(addr.into()).await.unwrap();
|
let io = conn.call(addr.into()).await.unwrap();
|
||||||
assert_eq!(io.query::<PeerAddr>().get().unwrap(), srv.addr().into());
|
assert_eq!(io.query::<PeerAddr>().get().unwrap(), srv.addr().into());
|
||||||
|
@ -140,7 +140,7 @@ async fn test_openssl_read_before_error() {
|
||||||
let mut builder = SslConnector::builder(SslMethod::tls()).unwrap();
|
let mut builder = SslConnector::builder(SslMethod::tls()).unwrap();
|
||||||
builder.set_verify(SslVerifyMode::NONE);
|
builder.set_verify(SslVerifyMode::NONE);
|
||||||
|
|
||||||
let conn = ntex::connect::openssl::Connector::new(builder.build());
|
let conn = Container::new(ntex::connect::openssl::Connector::new(builder.build()));
|
||||||
let addr = format!("127.0.0.1:{}", srv.addr().port());
|
let addr = format!("127.0.0.1:{}", srv.addr().port());
|
||||||
let io = conn.call(addr.into()).await.unwrap();
|
let io = conn.call(addr.into()).await.unwrap();
|
||||||
let item = io.recv(&Rc::new(BytesCodec)).await.unwrap().unwrap();
|
let item = io.recv(&Rc::new(BytesCodec)).await.unwrap().unwrap();
|
||||||
|
@ -185,7 +185,7 @@ async fn test_rustls_string() {
|
||||||
.with_custom_certificate_verifier(Arc::new(danger::NoCertificateVerification {}))
|
.with_custom_certificate_verifier(Arc::new(danger::NoCertificateVerification {}))
|
||||||
.with_no_client_auth();
|
.with_no_client_auth();
|
||||||
|
|
||||||
let conn = ntex::connect::rustls::Connector::new(config);
|
let conn = Container::new(ntex::connect::rustls::Connector::new(config));
|
||||||
let addr = format!("localhost:{}", srv.addr().port());
|
let addr = format!("localhost:{}", srv.addr().port());
|
||||||
let io = conn.call(addr.into()).await.unwrap();
|
let io = conn.call(addr.into()).await.unwrap();
|
||||||
assert_eq!(io.query::<PeerAddr>().get().unwrap(), srv.addr().into());
|
assert_eq!(io.query::<PeerAddr>().get().unwrap(), srv.addr().into());
|
||||||
|
@ -225,13 +225,13 @@ async fn test_static_str() {
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
|
|
||||||
let conn = ntex::connect::Connector::new();
|
let conn = Container::new(ntex::connect::Connector::new());
|
||||||
|
|
||||||
let io = conn.call(Connect::with("10", srv.addr())).await.unwrap();
|
let io = conn.call(Connect::with("10", srv.addr())).await.unwrap();
|
||||||
assert_eq!(io.query::<PeerAddr>().get().unwrap(), srv.addr().into());
|
assert_eq!(io.query::<PeerAddr>().get().unwrap(), srv.addr().into());
|
||||||
|
|
||||||
let connect = Connect::new("127.0.0.1".to_owned());
|
let connect = Connect::new("127.0.0.1".to_owned());
|
||||||
let conn = ntex::connect::Connector::new();
|
let conn = Container::new(ntex::connect::Connector::new());
|
||||||
let io = conn.call(connect).await;
|
let io = conn.call(connect).await;
|
||||||
assert!(io.is_err());
|
assert!(io.is_err());
|
||||||
}
|
}
|
||||||
|
@ -248,7 +248,7 @@ async fn test_create() {
|
||||||
});
|
});
|
||||||
|
|
||||||
let factory = ntex::connect::Connector::new();
|
let factory = ntex::connect::Connector::new();
|
||||||
let conn = factory.create(()).await.unwrap();
|
let conn = factory.container(()).await.unwrap();
|
||||||
let io = conn.call(Connect::with("10", srv.addr())).await.unwrap();
|
let io = conn.call(Connect::with("10", srv.addr())).await.unwrap();
|
||||||
assert_eq!(io.query::<PeerAddr>().get().unwrap(), srv.addr().into());
|
assert_eq!(io.query::<PeerAddr>().get().unwrap(), srv.addr().into());
|
||||||
}
|
}
|
||||||
|
@ -265,7 +265,7 @@ async fn test_uri() {
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
|
|
||||||
let conn = ntex::connect::Connector::default();
|
let conn = Container::new(ntex::connect::Connector::default());
|
||||||
let addr =
|
let addr =
|
||||||
ntex::http::Uri::try_from(format!("https://localhost:{}", srv.addr().port()))
|
ntex::http::Uri::try_from(format!("https://localhost:{}", srv.addr().port()))
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -285,7 +285,7 @@ async fn test_rustls_uri() {
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
|
|
||||||
let conn = ntex::connect::Connector::default();
|
let conn = Container::new(ntex::connect::Connector::default());
|
||||||
let addr =
|
let addr =
|
||||||
ntex::http::Uri::try_from(format!("https://localhost:{}", srv.addr().port()))
|
ntex::http::Uri::try_from(format!("https://localhost:{}", srv.addr().port()))
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
|
@ -4,7 +4,7 @@ use ntex::codec::BytesCodec;
|
||||||
use ntex::http::test::server as test_server;
|
use ntex::http::test::server as test_server;
|
||||||
use ntex::http::{body, h1, test, HttpService, Request, Response, StatusCode};
|
use ntex::http::{body, h1, test, HttpService, Request, Response, StatusCode};
|
||||||
use ntex::io::{DispatchItem, Dispatcher, Io};
|
use ntex::io::{DispatchItem, Dispatcher, Io};
|
||||||
use ntex::service::{fn_factory, Service};
|
use ntex::service::{fn_factory, Ctx, Service};
|
||||||
use ntex::util::{BoxFuture, ByteString, Bytes, Ready};
|
use ntex::util::{BoxFuture, ByteString, Bytes, Ready};
|
||||||
use ntex::ws::{self, handshake, handshake_response};
|
use ntex::ws::{self, handshake, handshake_response};
|
||||||
|
|
||||||
|
@ -40,7 +40,11 @@ impl Service<(Request, Io, h1::Codec)> for WsService {
|
||||||
Poll::Ready(Ok(()))
|
Poll::Ready(Ok(()))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call(&self, (req, io, codec): (Request, Io, h1::Codec)) -> Self::Future<'_> {
|
fn call<'a>(
|
||||||
|
&'a self,
|
||||||
|
(req, io, codec): (Request, Io, h1::Codec),
|
||||||
|
_: Ctx<'a, Self>,
|
||||||
|
) -> Self::Future<'a> {
|
||||||
let fut = async move {
|
let fut = async move {
|
||||||
let res = handshake(req.head()).unwrap().message_body(());
|
let res = handshake(req.head()).unwrap().message_body(());
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue