mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-04 13:27:39 +03:00
commit
4714064872
25 changed files with 166 additions and 157 deletions
|
@ -34,5 +34,3 @@ 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" }
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
## [0.3.0] - 2023-06-xx
|
## [0.3.0-beta.0] - 2023-06-16
|
||||||
|
|
||||||
* Migrate to ntex-service 1.2
|
* Migrate to ntex-service 1.2
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-async-std"
|
name = "ntex-async-std"
|
||||||
version = "0.3.0"
|
version = "0.3.0-beta.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.3.0"
|
ntex-io = "0.3.0-beta.0"
|
||||||
ntex-util = "0.3.0"
|
ntex-util = "0.3.0-beta.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.7.0", features = ["tokio"] }
|
ntex = { version = "0.7.0-beta.0", features = ["tokio"] }
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
## [0.3.0] - 2023-06-xx
|
## [0.3.0-beta.0] - 2023-06-16
|
||||||
|
|
||||||
* Migrate to ntex-service 1.2
|
* Migrate to ntex-service 1.2
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-connect"
|
name = "ntex-connect"
|
||||||
version = "0.3.0"
|
version = "0.3.0-beta.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,19 +34,19 @@ 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.2.0"
|
ntex-service = "1.2.0-beta.0"
|
||||||
ntex-bytes = "0.1.19"
|
ntex-bytes = "0.1.19"
|
||||||
ntex-http = "0.1.8"
|
ntex-http = "0.1.8"
|
||||||
ntex-io = "0.3.0"
|
ntex-io = "0.3.0-beta.0"
|
||||||
ntex-rt = "0.4.7"
|
ntex-rt = "0.4.7"
|
||||||
ntex-tls = "0.3.0"
|
ntex-tls = "0.3.0-beta.0"
|
||||||
ntex-util = "0.3.0"
|
ntex-util = "0.3.0-beta.0"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
thiserror = "1.0"
|
thiserror = "1.0"
|
||||||
|
|
||||||
ntex-tokio = { version = "0.3.0", optional = true }
|
ntex-tokio = { version = "0.3.0-beta.0", optional = true }
|
||||||
ntex-glommio = { version = "0.3.0", optional = true }
|
ntex-glommio = { version = "0.3.0-beta.0", optional = true }
|
||||||
ntex-async-std = { version = "0.3.0", optional = true }
|
ntex-async-std = { version = "0.3.0-beta.0", optional = true }
|
||||||
|
|
||||||
# openssl
|
# openssl
|
||||||
tls-openssl = { version="0.10", package = "openssl", optional = true }
|
tls-openssl = { version="0.10", package = "openssl", optional = true }
|
||||||
|
@ -58,4 +58,4 @@ 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.7.0", features = ["tokio"] }
|
ntex = { version = "0.7.0-beta.0", features = ["tokio"] }
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
## [0.3.0] - 2023-06-xx
|
## [0.3.0-beta.0] - 2023-06-16
|
||||||
|
|
||||||
* Migrate to ntex-service 1.2
|
* Migrate to ntex-service 1.2
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-glommio"
|
name = "ntex-glommio"
|
||||||
version = "0.3.0"
|
version = "0.3.0-beta.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.3.0"
|
ntex-io = "0.3.0-beta.0"
|
||||||
ntex-util = "0.3.0"
|
ntex-util = "0.3.0-beta.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,6 +1,6 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
## [0.3.0] - 2023-06-xx
|
## [0.3.0-beta.0] - 2023-06-16
|
||||||
|
|
||||||
* Migrate to ntex-service 1.2
|
* Migrate to ntex-service 1.2
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-io"
|
name = "ntex-io"
|
||||||
version = "0.3.0"
|
version = "0.3.0-beta.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.3.0"
|
ntex-util = "0.3.0-beta.0"
|
||||||
ntex-service = "1.2.0"
|
ntex-service = "1.2.0-beta.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.7.0", features = ["tokio"] }
|
ntex = { version = "0.7.0-beta.0", features = ["tokio"] }
|
||||||
|
|
|
@ -16,6 +16,6 @@ syn = { version = "^1", features = ["full", "parsing"] }
|
||||||
proc-macro2 = "^1"
|
proc-macro2 = "^1"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
ntex = { version = "0.7.0", features = ["tokio"] }
|
ntex = { version = "0.7.0-beta.0", features = ["tokio"] }
|
||||||
futures = "0.3"
|
futures = "0.3"
|
||||||
env_logger = "0.10"
|
env_logger = "0.10"
|
|
@ -1,6 +1,6 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
## [1.2.0] - 2023-06-xx
|
## [1.2.0-beta.0] - 2023-06-16
|
||||||
|
|
||||||
* Enforce service readiness during call
|
* Enforce service readiness during call
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-service"
|
name = "ntex-service"
|
||||||
version = "1.2.0"
|
version = "1.2.0-beta.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "ntex service"
|
description = "ntex service"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -20,5 +20,5 @@ pin-project-lite = "0.2.6"
|
||||||
slab = "0.4"
|
slab = "0.4"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
ntex = { version = "0.7.0", features = ["tokio"] }
|
ntex = { version = "0.7.0-beta.0", features = ["tokio"] }
|
||||||
ntex-util = "0.3.0"
|
ntex-util = "0.3.0-beta.0"
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
#![allow(clippy::type_complexity)]
|
#![allow(clippy::type_complexity)]
|
||||||
use std::{cell::RefCell, future::Future, marker, pin::Pin, rc::Rc, task, task::Poll};
|
use std::{future::Future, marker, pin::Pin, rc::Rc, task, task::Poll};
|
||||||
|
|
||||||
use super::{Ctx, IntoService, IntoServiceFactory, Service, ServiceCall, ServiceFactory};
|
use super::ctx::{Ctx, ServiceCall, Waiters};
|
||||||
|
use super::{IntoService, IntoServiceFactory, Service, ServiceFactory};
|
||||||
|
|
||||||
/// Apply transform function to a service.
|
/// Apply transform function to a service.
|
||||||
pub fn apply_fn<T, Req, F, R, In, Out, Err, U>(
|
pub fn apply_fn<T, Req, F, R, In, Out, Err, U>(
|
||||||
|
@ -10,7 +11,7 @@ pub fn apply_fn<T, Req, F, R, In, Out, Err, U>(
|
||||||
) -> Apply<T, Req, F, R, In, Out, Err>
|
) -> Apply<T, Req, F, R, In, Out, Err>
|
||||||
where
|
where
|
||||||
T: Service<Req, Error = Err>,
|
T: Service<Req, Error = Err>,
|
||||||
F: Fn(In, ApplyService<T>) -> R,
|
for<'r> F: Fn(In, ApplyService<T>) -> R,
|
||||||
R: Future<Output = Result<Out, Err>>,
|
R: Future<Output = Result<Out, Err>>,
|
||||||
U: IntoService<T, Req>,
|
U: IntoService<T, Req>,
|
||||||
{
|
{
|
||||||
|
@ -74,8 +75,7 @@ where
|
||||||
|
|
||||||
pub struct ApplyService<S> {
|
pub struct ApplyService<S> {
|
||||||
svc: Rc<S>,
|
svc: Rc<S>,
|
||||||
index: usize,
|
waiters: Waiters,
|
||||||
waiters: Rc<RefCell<slab::Slab<Option<task::Waker>>>>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S> ApplyService<S> {
|
impl<S> ApplyService<S> {
|
||||||
|
@ -83,7 +83,7 @@ impl<S> ApplyService<S> {
|
||||||
where
|
where
|
||||||
S: Service<R>,
|
S: Service<R>,
|
||||||
{
|
{
|
||||||
Ctx::<S>::new(self.index, &self.waiters).call(&self.svc, req)
|
Ctx::<S>::new(&self.waiters).call(&self.svc, req)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -102,11 +102,9 @@ where
|
||||||
|
|
||||||
#[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> {
|
||||||
let (index, waiters) = ctx.into_inner();
|
|
||||||
let svc = ApplyService {
|
let svc = ApplyService {
|
||||||
index,
|
|
||||||
waiters: waiters.clone(),
|
|
||||||
svc: self.service.clone(),
|
svc: self.service.clone(),
|
||||||
|
waiters: ctx.waiters().clone(),
|
||||||
};
|
};
|
||||||
(self.f)(req, svc)
|
(self.f)(req, svc)
|
||||||
}
|
}
|
||||||
|
@ -161,7 +159,7 @@ impl<T, Req, Cfg, F, R, In, Out, Err> ServiceFactory<In, Cfg>
|
||||||
where
|
where
|
||||||
T: ServiceFactory<Req, Cfg, Error = Err>,
|
T: ServiceFactory<Req, Cfg, Error = Err>,
|
||||||
F: Fn(In, ApplyService<T::Service>) -> R + Clone,
|
F: Fn(In, ApplyService<T::Service>) -> R + Clone,
|
||||||
R: Future<Output = Result<Out, Err>>,
|
for<'r> R: Future<Output = Result<Out, Err>> + 'r,
|
||||||
{
|
{
|
||||||
type Response = Out;
|
type Response = Out;
|
||||||
type Error = Err;
|
type Error = Err;
|
||||||
|
@ -186,6 +184,7 @@ pin_project_lite::pin_project! {
|
||||||
T: ServiceFactory<Req, Cfg, Error = Err>,
|
T: ServiceFactory<Req, Cfg, Error = Err>,
|
||||||
T: 'f,
|
T: 'f,
|
||||||
F: Fn(In, ApplyService<T::Service>) -> R,
|
F: Fn(In, ApplyService<T::Service>) -> R,
|
||||||
|
T::Service: 'f,
|
||||||
R: Future<Output = Result<Out, Err>>,
|
R: Future<Output = Result<Out, Err>>,
|
||||||
Cfg: 'f,
|
Cfg: 'f,
|
||||||
{
|
{
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
use std::task::{Context, Poll, Waker};
|
use std::{future::Future, pin::Pin, task::Context, task::Poll};
|
||||||
use std::{cell::RefCell, future::Future, pin::Pin, rc::Rc};
|
|
||||||
|
|
||||||
use crate::Ctx;
|
use crate::ctx::{Ctx, Waiters};
|
||||||
|
|
||||||
pub type BoxFuture<'a, I, E> = Pin<Box<dyn Future<Output = Result<I, E>> + 'a>>;
|
pub type BoxFuture<'a, I, E> = Pin<Box<dyn Future<Output = Result<I, E>> + 'a>>;
|
||||||
|
|
||||||
|
@ -44,8 +43,7 @@ trait ServiceObj<Req> {
|
||||||
fn call<'a>(
|
fn call<'a>(
|
||||||
&'a self,
|
&'a self,
|
||||||
req: Req,
|
req: Req,
|
||||||
idx: usize,
|
waiters: &'a Waiters,
|
||||||
waiters: &'a Rc<RefCell<slab::Slab<Option<Waker>>>>,
|
|
||||||
) -> BoxFuture<'a, Self::Response, Self::Error>;
|
) -> BoxFuture<'a, Self::Response, Self::Error>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,10 +69,9 @@ where
|
||||||
fn call<'a>(
|
fn call<'a>(
|
||||||
&'a self,
|
&'a self,
|
||||||
req: Req,
|
req: Req,
|
||||||
idx: usize,
|
waiters: &'a Waiters,
|
||||||
waiters: &'a Rc<RefCell<slab::Slab<Option<Waker>>>>,
|
|
||||||
) -> BoxFuture<'a, Self::Response, Self::Error> {
|
) -> BoxFuture<'a, Self::Response, Self::Error> {
|
||||||
Box::pin(Ctx::<'a, S>::new(idx, waiters).call_nowait(self, req))
|
Box::pin(Ctx::<'a, S>::new(waiters).call_nowait(self, req))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -135,8 +132,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> {
|
||||||
let (index, waiters) = ctx.into_inner();
|
self.0.call(req, ctx.waiters())
|
||||||
self.0.call(req, index, waiters)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,22 +1,75 @@
|
||||||
use std::{cell::RefCell, future::Future, marker, ops, pin::Pin, rc::Rc, task, task::Poll};
|
use std::{
|
||||||
|
cell::UnsafeCell, future::Future, marker, ops, pin::Pin, rc::Rc, task, task::Poll,
|
||||||
|
};
|
||||||
|
|
||||||
use crate::{Service, ServiceFactory};
|
use crate::{Service, ServiceFactory};
|
||||||
|
|
||||||
|
/// Container for a service.
|
||||||
|
///
|
||||||
|
/// Container allows to call enclosed service and adds support of shared readiness.
|
||||||
pub struct Container<S> {
|
pub struct Container<S> {
|
||||||
svc: Rc<S>,
|
svc: Rc<S>,
|
||||||
|
waiters: Waiters,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Ctx<'a, S: ?Sized> {
|
||||||
|
waiters: &'a Waiters,
|
||||||
|
_t: marker::PhantomData<Rc<S>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) struct Waiters {
|
||||||
index: usize,
|
index: usize,
|
||||||
waiters: Rc<RefCell<slab::Slab<Option<task::Waker>>>>,
|
waiters: Rc<UnsafeCell<slab::Slab<Option<task::Waker>>>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Waiters {
|
||||||
|
#[allow(clippy::mut_from_ref)]
|
||||||
|
fn get(&self) -> &mut slab::Slab<Option<task::Waker>> {
|
||||||
|
unsafe { &mut *self.waiters.as_ref().get() }
|
||||||
|
}
|
||||||
|
|
||||||
|
fn notify(&self) {
|
||||||
|
for (_, waker) in self.get().iter_mut() {
|
||||||
|
if let Some(waker) = waker.take() {
|
||||||
|
waker.wake();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn register(&self, cx: &mut task::Context<'_>) {
|
||||||
|
self.get()[self.index] = Some(cx.waker().clone());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Clone for Waiters {
|
||||||
|
fn clone(&self) -> Self {
|
||||||
|
Waiters {
|
||||||
|
index: self.get().insert(None),
|
||||||
|
waiters: self.waiters.clone(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for Waiters {
|
||||||
|
#[inline]
|
||||||
|
fn drop(&mut self) {
|
||||||
|
self.get().remove(self.index);
|
||||||
|
self.notify();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S> Container<S> {
|
impl<S> Container<S> {
|
||||||
#[inline]
|
#[inline]
|
||||||
|
/// Construct new container instance.
|
||||||
pub fn new(svc: S) -> Self {
|
pub fn new(svc: S) -> Self {
|
||||||
let mut waiters = slab::Slab::new();
|
let mut waiters = slab::Slab::new();
|
||||||
let index = waiters.insert(None);
|
let index = waiters.insert(None);
|
||||||
Container {
|
Container {
|
||||||
index,
|
|
||||||
svc: Rc::new(svc),
|
svc: Rc::new(svc),
|
||||||
waiters: Rc::new(RefCell::new(waiters)),
|
waiters: Waiters {
|
||||||
|
index,
|
||||||
|
waiters: Rc::new(UnsafeCell::new(waiters)),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -27,9 +80,10 @@ impl<S> Container<S> {
|
||||||
S: Service<R>,
|
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() {
|
||||||
self.waiters.borrow_mut()[self.index] = Some(cx.waker().clone());
|
self.waiters.register(cx)
|
||||||
|
} else {
|
||||||
|
self.waiters.notify()
|
||||||
}
|
}
|
||||||
res
|
res
|
||||||
}
|
}
|
||||||
|
@ -50,7 +104,6 @@ impl<S> Container<S> {
|
||||||
S: Service<R>,
|
S: Service<R>,
|
||||||
{
|
{
|
||||||
let ctx = Ctx::<'a, S> {
|
let ctx = Ctx::<'a, S> {
|
||||||
index: self.index,
|
|
||||||
waiters: &self.waiters,
|
waiters: &self.waiters,
|
||||||
_t: marker::PhantomData,
|
_t: marker::PhantomData,
|
||||||
};
|
};
|
||||||
|
@ -61,12 +114,10 @@ impl<S> Container<S> {
|
||||||
f: &F,
|
f: &F,
|
||||||
cfg: C,
|
cfg: C,
|
||||||
) -> ContainerFactory<'_, F, R, C> {
|
) -> ContainerFactory<'_, F, R, C> {
|
||||||
ContainerFactory {
|
ContainerFactory { fut: f.create(cfg) }
|
||||||
fut: f.create(cfg),
|
|
||||||
_t: marker::PhantomData,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Extract service if container hadnt been cloned before.
|
||||||
pub fn into_service(self) -> Option<S> {
|
pub fn into_service(self) -> Option<S> {
|
||||||
let svc = self.svc.clone();
|
let svc = self.svc.clone();
|
||||||
drop(self);
|
drop(self);
|
||||||
|
@ -75,11 +126,9 @@ impl<S> Container<S> {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S> Clone for Container<S> {
|
impl<S> Clone for Container<S> {
|
||||||
|
#[inline]
|
||||||
fn clone(&self) -> Self {
|
fn clone(&self) -> Self {
|
||||||
let index = self.waiters.borrow_mut().insert(None);
|
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
index,
|
|
||||||
svc: self.svc.clone(),
|
svc: self.svc.clone(),
|
||||||
waiters: self.waiters.clone(),
|
waiters: self.waiters.clone(),
|
||||||
}
|
}
|
||||||
|
@ -87,6 +136,7 @@ impl<S> Clone for Container<S> {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S> From<S> for Container<S> {
|
impl<S> From<S> for Container<S> {
|
||||||
|
#[inline]
|
||||||
fn from(svc: S) -> Self {
|
fn from(svc: S) -> Self {
|
||||||
Container::new(svc)
|
Container::new(svc)
|
||||||
}
|
}
|
||||||
|
@ -101,41 +151,16 @@ impl<S> ops::Deref for Container<S> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S> Drop for Container<S> {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
let mut waiters = self.waiters.borrow_mut();
|
|
||||||
|
|
||||||
waiters.remove(self.index);
|
|
||||||
for (_, waker) in &mut *waiters {
|
|
||||||
if let Some(waker) = waker.take() {
|
|
||||||
waker.wake();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Ctx<'a, S: ?Sized> {
|
|
||||||
index: usize,
|
|
||||||
waiters: &'a Rc<RefCell<slab::Slab<Option<task::Waker>>>>,
|
|
||||||
_t: marker::PhantomData<Rc<S>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, S: ?Sized> Ctx<'a, S> {
|
impl<'a, S: ?Sized> Ctx<'a, S> {
|
||||||
pub(crate) fn new(
|
pub(crate) fn new(waiters: &'a Waiters) -> Self {
|
||||||
index: usize,
|
|
||||||
waiters: &'a Rc<RefCell<slab::Slab<Option<task::Waker>>>>,
|
|
||||||
) -> Self {
|
|
||||||
Self {
|
Self {
|
||||||
index,
|
|
||||||
waiters,
|
waiters,
|
||||||
_t: marker::PhantomData,
|
_t: marker::PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn into_inner(
|
pub(crate) fn waiters(self) -> &'a Waiters {
|
||||||
self,
|
self.waiters
|
||||||
) -> (usize, &'a Rc<RefCell<slab::Slab<Option<task::Waker>>>>) {
|
|
||||||
(self.index, self.waiters)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Call service, do not check service readiness
|
/// Call service, do not check service readiness
|
||||||
|
@ -147,7 +172,6 @@ impl<'a, S: ?Sized> Ctx<'a, S> {
|
||||||
svc.call(
|
svc.call(
|
||||||
req,
|
req,
|
||||||
Ctx {
|
Ctx {
|
||||||
index: self.index,
|
|
||||||
waiters: self.waiters,
|
waiters: self.waiters,
|
||||||
_t: marker::PhantomData,
|
_t: marker::PhantomData,
|
||||||
},
|
},
|
||||||
|
@ -165,7 +189,6 @@ impl<'a, S: ?Sized> Ctx<'a, S> {
|
||||||
state: ServiceCallState::Ready {
|
state: ServiceCallState::Ready {
|
||||||
svc,
|
svc,
|
||||||
req: Some(req),
|
req: Some(req),
|
||||||
index: self.index,
|
|
||||||
waiters: self.waiters,
|
waiters: self.waiters,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -175,9 +198,9 @@ impl<'a, S: ?Sized> Ctx<'a, S> {
|
||||||
impl<'a, S: ?Sized> Copy for Ctx<'a, S> {}
|
impl<'a, S: ?Sized> Copy for Ctx<'a, S> {}
|
||||||
|
|
||||||
impl<'a, S: ?Sized> Clone for Ctx<'a, S> {
|
impl<'a, S: ?Sized> Clone for Ctx<'a, S> {
|
||||||
|
#[inline]
|
||||||
fn clone(&self) -> Self {
|
fn clone(&self) -> Self {
|
||||||
Self {
|
Self {
|
||||||
index: self.index,
|
|
||||||
waiters: self.waiters,
|
waiters: self.waiters,
|
||||||
_t: marker::PhantomData,
|
_t: marker::PhantomData,
|
||||||
}
|
}
|
||||||
|
@ -209,8 +232,7 @@ pin_project_lite::pin_project! {
|
||||||
{
|
{
|
||||||
Ready { req: Option<Req>,
|
Ready { req: Option<Req>,
|
||||||
svc: &'a T,
|
svc: &'a T,
|
||||||
index: usize,
|
waiters: &'a Waiters,
|
||||||
waiters: &'a Rc<RefCell<slab::Slab<Option<task::Waker>>>>,
|
|
||||||
},
|
},
|
||||||
Call { #[pin] fut: T::Future<'a> },
|
Call { #[pin] fut: T::Future<'a> },
|
||||||
Empty,
|
Empty,
|
||||||
|
@ -227,35 +249,27 @@ where
|
||||||
let mut this = self.as_mut().project();
|
let mut this = self.as_mut().project();
|
||||||
|
|
||||||
match this.state.as_mut().project() {
|
match this.state.as_mut().project() {
|
||||||
ServiceCallStateProject::Ready {
|
ServiceCallStateProject::Ready { req, svc, waiters } => {
|
||||||
req,
|
match svc.poll_ready(cx)? {
|
||||||
svc,
|
Poll::Ready(()) => {
|
||||||
index,
|
waiters.notify();
|
||||||
waiters,
|
|
||||||
} => match svc.poll_ready(cx)? {
|
|
||||||
Poll::Ready(()) => {
|
|
||||||
for (_, waker) in &mut *waiters.borrow_mut() {
|
|
||||||
if let Some(waker) = waker.take() {
|
|
||||||
waker.wake();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let fut = svc.call(
|
let fut = svc.call(
|
||||||
req.take().unwrap(),
|
req.take().unwrap(),
|
||||||
Ctx {
|
Ctx {
|
||||||
waiters,
|
waiters,
|
||||||
index: *index,
|
_t: marker::PhantomData,
|
||||||
_t: marker::PhantomData,
|
},
|
||||||
},
|
);
|
||||||
);
|
this.state.set(ServiceCallState::Call { fut });
|
||||||
this.state.set(ServiceCallState::Call { fut });
|
self.poll(cx)
|
||||||
self.poll(cx)
|
}
|
||||||
|
Poll::Pending => {
|
||||||
|
waiters.register(cx);
|
||||||
|
Poll::Pending
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Poll::Pending => {
|
}
|
||||||
waiters.borrow_mut()[*index] = Some(cx.waker().clone());
|
|
||||||
Poll::Pending
|
|
||||||
}
|
|
||||||
},
|
|
||||||
ServiceCallStateProject::Call { fut } => fut.poll(cx).map(|r| {
|
ServiceCallStateProject::Call { fut } => fut.poll(cx).map(|r| {
|
||||||
this.state.set(ServiceCallState::Empty);
|
this.state.set(ServiceCallState::Empty);
|
||||||
r
|
r
|
||||||
|
@ -277,7 +291,6 @@ pin_project_lite::pin_project! {
|
||||||
{
|
{
|
||||||
#[pin]
|
#[pin]
|
||||||
fut: F::Future<'f>,
|
fut: F::Future<'f>,
|
||||||
_t: marker::PhantomData<(R, C)>,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -80,6 +80,10 @@ pub use self::pipeline::{pipeline, pipeline_factory, Pipeline, PipelineFactory};
|
||||||
/// ```rust,ignore
|
/// ```rust,ignore
|
||||||
/// async fn my_service(req: u8) -> Result<u64, Infallible>;
|
/// async fn my_service(req: u8) -> Result<u64, Infallible>;
|
||||||
/// ```
|
/// ```
|
||||||
|
///
|
||||||
|
/// Service cannot be called directly, it must be wrapped to an instance of [`Container`] or
|
||||||
|
/// by using `ctx` argument of the call method in case of chanined services.
|
||||||
|
///
|
||||||
pub trait Service<Req> {
|
pub trait Service<Req> {
|
||||||
/// Responses given by the service.
|
/// Responses given by the service.
|
||||||
type Response;
|
type Response;
|
||||||
|
@ -96,11 +100,9 @@ pub trait Service<Req> {
|
||||||
/// Process the request and return the response asynchronously.
|
/// Process the request and return the response asynchronously.
|
||||||
///
|
///
|
||||||
/// This function is expected to be callable off-task. As such, implementations of `call`
|
/// This function is expected to be callable off-task. As such, implementations of `call`
|
||||||
/// should take care to not call `poll_ready`. If the service is at capacity and the request
|
/// should take care to not call `poll_ready`. Caller of the service verifies readiness,
|
||||||
/// is unable to be handled, the returned `Future` should resolve to an error.
|
/// Only way to make a `call` is to use `ctx` argument, it enforces readiness before calling
|
||||||
///
|
/// service.
|
||||||
/// Invoking `call` without first invoking `poll_ready` is permitted. Implementations must be
|
|
||||||
/// 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>;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
@ -116,7 +118,8 @@ pub trait Service<Req> {
|
||||||
/// # Notes
|
/// # Notes
|
||||||
///
|
///
|
||||||
/// 1. `.poll_ready()` might be called on different task from actual service call.
|
/// 1. `.poll_ready()` might be called on different task from actual service call.
|
||||||
/// 1. In case of chained services, `.poll_ready()` is called for all services at once.
|
/// 2. In case of chained services, `.poll_ready()` is called for all services at once.
|
||||||
|
/// 3. Every `.call()` in chained services enforces readiness.
|
||||||
fn poll_ready(&self, cx: &mut task::Context<'_>) -> Poll<Result<(), Self::Error>> {
|
fn poll_ready(&self, cx: &mut task::Context<'_>) -> Poll<Result<(), Self::Error>> {
|
||||||
Poll::Ready(Ok(()))
|
Poll::Ready(Ok(()))
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
## [0.3.0] - 2023-06-xx
|
## [0.3.0-beta.0] - 2023-06-16
|
||||||
|
|
||||||
* Migrate to ntex-service 1.2
|
* Migrate to ntex-service 1.2
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-tls"
|
name = "ntex-tls"
|
||||||
version = "0.3.0"
|
version = "0.3.0-beta.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,9 +26,9 @@ rustls = ["tls_rust"]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-bytes = "0.1.19"
|
ntex-bytes = "0.1.19"
|
||||||
ntex-io = "0.3.0"
|
ntex-io = "0.3.0-beta.0"
|
||||||
ntex-util = "0.3.0"
|
ntex-util = "0.3.0-beta.0"
|
||||||
ntex-service = "1.2.0"
|
ntex-service = "1.2.0-beta.0"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
pin-project-lite = "0.2"
|
pin-project-lite = "0.2"
|
||||||
|
|
||||||
|
@ -39,7 +39,7 @@ tls_openssl = { version = "0.10", package = "openssl", optional = true }
|
||||||
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.7.0", features = ["openssl", "rustls", "tokio"] }
|
ntex = { version = "0.7.0-beta.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.23" }
|
webpki-roots = { version = "0.23" }
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
## [0.3.0] - 2023-06-xx
|
## [0.3.0-beta.0] - 2023-06-16
|
||||||
|
|
||||||
* Migrate to ntex-service 1.2
|
* Migrate to ntex-service 1.2
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-tokio"
|
name = "ntex-tokio"
|
||||||
version = "0.3.0"
|
version = "0.3.0-beta.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.3.0"
|
ntex-io = "0.3.0-beta.0"
|
||||||
ntex-util = "0.3.0"
|
ntex-util = "0.3.0-beta.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,6 +1,6 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
## [0.3.0] - 2023-06-xx
|
## [0.3.0-beta.0] - 2023-06-16
|
||||||
|
|
||||||
* Upgrade to ntex-service 1.2
|
* Upgrade to ntex-service 1.2
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-util"
|
name = "ntex-util"
|
||||||
version = "0.3.0"
|
version = "0.3.0-beta.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.2.0"
|
ntex-service = "1.2.0-beta.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.7.0", features = ["tokio"] }
|
ntex = { version = "0.7.0-beta.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,6 +1,6 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
## [0.7.0] - 2023-06-xx
|
## [0.7.0-beta.0] - 2023-06-16
|
||||||
|
|
||||||
* Migrate to ntex-service 1.2
|
* Migrate to ntex-service 1.2
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex"
|
name = "ntex"
|
||||||
version = "0.7.0"
|
version = "0.7.0-beta.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.3.0"
|
ntex-connect = "0.3.0-beta.0"
|
||||||
ntex-http = "0.1.9"
|
ntex-http = "0.1.9"
|
||||||
ntex-router = "0.5.1"
|
ntex-router = "0.5.1"
|
||||||
ntex-service = "1.2.0"
|
ntex-service = "1.2.0-beta.0"
|
||||||
ntex-macros = "0.1.3"
|
ntex-macros = "0.1.3"
|
||||||
ntex-util = "0.3.0"
|
ntex-util = "0.3.0-beta.0"
|
||||||
ntex-bytes = "0.1.19"
|
ntex-bytes = "0.1.19"
|
||||||
ntex-h2 = "0.3.0"
|
ntex-h2 = "0.3.0-beta.0"
|
||||||
ntex-rt = "0.4.9"
|
ntex-rt = "0.4.9"
|
||||||
ntex-io = "0.3.0"
|
ntex-io = "0.3.0-beta.0"
|
||||||
ntex-tls = "0.3.0"
|
ntex-tls = "0.3.0-beta.0"
|
||||||
ntex-tokio = { version = "0.3.0", optional = true }
|
ntex-tokio = { version = "0.3.0-beta.0", optional = true }
|
||||||
ntex-glommio = { version = "0.3.0", optional = true }
|
ntex-glommio = { version = "0.3.0-beta.0", optional = true }
|
||||||
ntex-async-std = { version = "0.3.0", optional = true }
|
ntex-async-std = { version = "0.3.0-beta.0", optional = true }
|
||||||
|
|
||||||
async-oneshot = "0.5.0"
|
async-oneshot = "0.5.0"
|
||||||
async-channel = "1.8.0"
|
async-channel = "1.8.0"
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue