Add .on_shutdown() callback to fn_service

This commit is contained in:
Nikolay Kim 2021-03-26 20:11:07 +06:00
parent 91d43c0053
commit 290f863976
3 changed files with 129 additions and 20 deletions

View file

@ -1,5 +1,9 @@
# Changes # Changes
## [0.1.6] - 2021-03-26
* Add .on_shutdown() callback to fn_service
## [0.1.5] - 2021-01-13 ## [0.1.5] - 2021-01-13
* Use pin-project-lite instead of pin-project * Use pin-project-lite instead of pin-project

View file

@ -1,6 +1,6 @@
[package] [package]
name = "ntex-service" name = "ntex-service"
version = "0.1.5" version = "0.1.6"
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"]

View file

@ -1,4 +1,4 @@
use std::cell::RefCell; use std::cell::{Cell, RefCell};
use std::future::Future; use std::future::Future;
use std::marker::PhantomData; use std::marker::PhantomData;
use std::task::{Context, Poll}; use std::task::{Context, Poll};
@ -119,12 +119,16 @@ where
FnServiceConfig::new(f) FnServiceConfig::new(f)
} }
pub struct FnService<F, Fut, Req, Res, Err> #[inline]
pub fn fn_shutdown() {}
pub struct FnService<F, Fut, Req, Res, Err, FShut = fn()>
where where
F: Fn(Req) -> Fut, F: Fn(Req) -> Fut,
Fut: Future<Output = Result<Res, Err>>, Fut: Future<Output = Result<Res, Err>>,
{ {
f: F, f: F,
f_shutdown: Cell<Option<FShut>>,
_t: PhantomData<Req>, _t: PhantomData<Req>,
} }
@ -134,24 +138,49 @@ where
Fut: Future<Output = Result<Res, Err>>, Fut: Future<Output = Result<Res, Err>>,
{ {
pub(crate) fn new(f: F) -> Self { pub(crate) fn new(f: F) -> Self {
Self { f, _t: PhantomData } Self {
f,
f_shutdown: Cell::new(Some(fn_shutdown)),
_t: PhantomData,
}
}
/// Set function that get called oin poll_shutdown method of Service trait.
pub fn on_shutdown<FShut>(self, f: FShut) -> FnService<F, Fut, Req, Res, Err, FShut>
where
FShut: FnOnce(),
{
FnService {
f: self.f,
f_shutdown: Cell::new(Some(f)),
_t: PhantomData,
}
} }
} }
impl<F, Fut, Req, Res, Err> Clone for FnService<F, Fut, Req, Res, Err> impl<F, Fut, Req, Res, Err, FShut> Clone for FnService<F, Fut, Req, Res, Err, FShut>
where where
F: Fn(Req) -> Fut + Clone, F: Fn(Req) -> Fut + Clone,
FShut: FnOnce() + Clone,
Fut: Future<Output = Result<Res, Err>>, Fut: Future<Output = Result<Res, Err>>,
{ {
#[inline] #[inline]
fn clone(&self) -> Self { fn clone(&self) -> Self {
Self::new(self.f.clone()) let f = self.f_shutdown.take();
self.f_shutdown.set(f.clone());
Self {
f: self.f.clone(),
f_shutdown: Cell::new(f),
_t: PhantomData,
}
} }
} }
impl<F, Fut, Req, Res, Err> Service for FnService<F, Fut, Req, Res, Err> impl<F, Fut, Req, Res, Err, FShut> Service for FnService<F, Fut, Req, Res, Err, FShut>
where where
F: Fn(Req) -> Fut, F: Fn(Req) -> Fut,
FShut: FnOnce(),
Fut: Future<Output = Result<Res, Err>>, Fut: Future<Output = Result<Res, Err>>,
{ {
type Request = Req; type Request = Req;
@ -164,6 +193,14 @@ where
Poll::Ready(Ok(())) Poll::Ready(Ok(()))
} }
#[inline]
fn poll_shutdown(&self, _: &mut Context<'_>, _: bool) -> Poll<()> {
if let Some(f) = self.f_shutdown.take() {
(f)()
}
Poll::Ready(())
}
#[inline] #[inline]
fn call(&self, req: Req) -> Self::Future { fn call(&self, req: Req) -> Self::Future {
(self.f)(req) (self.f)(req)
@ -181,12 +218,13 @@ where
} }
} }
pub struct FnServiceFactory<F, Fut, Req, Res, Err, Cfg> pub struct FnServiceFactory<F, Fut, Req, Res, Err, Cfg, FShut = fn()>
where where
F: Fn(Req) -> Fut, F: Fn(Req) -> Fut,
Fut: Future<Output = Result<Res, Err>>, Fut: Future<Output = Result<Res, Err>>,
{ {
f: F, f: F,
f_shutdown: Cell<Option<FShut>>,
_t: PhantomData<(Req, Cfg)>, _t: PhantomData<(Req, Cfg)>,
} }
@ -196,24 +234,54 @@ where
Fut: Future<Output = Result<Res, Err>>, Fut: Future<Output = Result<Res, Err>>,
{ {
fn new(f: F) -> Self { fn new(f: F) -> Self {
FnServiceFactory { f, _t: PhantomData } FnServiceFactory {
f,
f_shutdown: Cell::new(Some(fn_shutdown)),
_t: PhantomData,
}
}
/// Set function that get called oin poll_shutdown method of Service trait.
pub fn on_shutdown<FShut>(
self,
f: FShut,
) -> FnServiceFactory<F, Fut, Req, Res, Err, Cfg, FShut>
where
FShut: FnOnce(),
{
FnServiceFactory {
f: self.f,
f_shutdown: Cell::new(Some(f)),
_t: PhantomData,
}
} }
} }
impl<F, Fut, Req, Res, Err, Cfg> Clone for FnServiceFactory<F, Fut, Req, Res, Err, Cfg> impl<F, Fut, Req, Res, Err, Cfg, FShut> Clone
for FnServiceFactory<F, Fut, Req, Res, Err, Cfg, FShut>
where where
F: Fn(Req) -> Fut + Clone, F: Fn(Req) -> Fut + Clone,
FShut: FnOnce() + Clone,
Fut: Future<Output = Result<Res, Err>>, Fut: Future<Output = Result<Res, Err>>,
{ {
#[inline] #[inline]
fn clone(&self) -> Self { fn clone(&self) -> Self {
Self::new(self.f.clone()) let f = self.f_shutdown.take();
self.f_shutdown.set(f.clone());
Self {
f: self.f.clone(),
f_shutdown: Cell::new(f),
_t: PhantomData,
}
} }
} }
impl<F, Fut, Req, Res, Err> Service for FnServiceFactory<F, Fut, Req, Res, Err, ()> impl<F, Fut, Req, Res, Err, FShut> Service
for FnServiceFactory<F, Fut, Req, Res, Err, (), FShut>
where where
F: Fn(Req) -> Fut + Clone, F: Fn(Req) -> Fut,
FShut: FnOnce(),
Fut: Future<Output = Result<Res, Err>>, Fut: Future<Output = Result<Res, Err>>,
{ {
type Request = Req; type Request = Req;
@ -226,16 +294,25 @@ where
Poll::Ready(Ok(())) Poll::Ready(Ok(()))
} }
#[inline]
fn poll_shutdown(&self, _: &mut Context<'_>, _: bool) -> Poll<()> {
if let Some(f) = self.f_shutdown.take() {
(f)()
}
Poll::Ready(())
}
#[inline] #[inline]
fn call(&self, req: Self::Request) -> Self::Future { fn call(&self, req: Self::Request) -> Self::Future {
(self.f)(req) (self.f)(req)
} }
} }
impl<F, Fut, Req, Res, Err, Cfg> ServiceFactory impl<F, Fut, Req, Res, Err, Cfg, FShut> ServiceFactory
for FnServiceFactory<F, Fut, Req, Res, Err, Cfg> for FnServiceFactory<F, Fut, Req, Res, Err, Cfg, FShut>
where where
F: Fn(Req) -> Fut + Clone, F: Fn(Req) -> Fut + Clone,
FShut: FnOnce() + Clone,
Fut: Future<Output = Result<Res, Err>>, Fut: Future<Output = Result<Res, Err>>,
{ {
type Request = Req; type Request = Req;
@ -243,13 +320,20 @@ where
type Error = Err; type Error = Err;
type Config = Cfg; type Config = Cfg;
type Service = FnService<F, Fut, Req, Res, Err>; type Service = FnService<F, Fut, Req, Res, Err, FShut>;
type InitError = (); type InitError = ();
type Future = Ready<Result<Self::Service, Self::InitError>>; type Future = Ready<Result<Self::Service, Self::InitError>>;
#[inline] #[inline]
fn new_service(&self, _: Cfg) -> Self::Future { fn new_service(&self, _: Cfg) -> Self::Future {
ok(FnService::new(self.f.clone())) let f = self.f_shutdown.take();
self.f_shutdown.set(f.clone());
ok(FnService {
f: self.f.clone(),
f_shutdown: Cell::new(f),
_t: PhantomData,
})
} }
} }
@ -445,7 +529,7 @@ where
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use std::task::Poll; use std::{rc::Rc, task::Poll};
use futures_util::future::{lazy, ok}; use futures_util::future::{lazy, ok};
@ -454,7 +538,12 @@ mod tests {
#[ntex::test] #[ntex::test]
async fn test_fn_service() { async fn test_fn_service() {
let new_srv = fn_service(|()| ok::<_, ()>("srv")).clone(); let shutdown = Rc::new(RefCell::new(false));
let new_srv = fn_service(|()| ok::<_, ()>("srv"))
.on_shutdown(|| {
*shutdown.borrow_mut() = true;
})
.clone();
let srv = new_srv.new_service(()).await.unwrap(); let srv = new_srv.new_service(()).await.unwrap();
let res = srv.call(()).await; let res = srv.call(()).await;
@ -466,6 +555,12 @@ mod tests {
let res = srv2.call(()).await; let res = srv2.call(()).await;
assert!(res.is_ok()); assert!(res.is_ok());
assert_eq!(res.unwrap(), "srv"); assert_eq!(res.unwrap(), "srv");
assert_eq!(
lazy(|cx| srv2.poll_shutdown(cx, false)).await,
Poll::Ready(())
);
assert!(*shutdown.borrow());
} }
#[ntex::test] #[ntex::test]
@ -480,12 +575,22 @@ mod tests {
#[ntex::test] #[ntex::test]
async fn test_fn_service_service() { async fn test_fn_service_service() {
let srv = fn_service(|()| ok::<_, ()>("srv")).clone(); let shutdown = Rc::new(RefCell::new(false));
let srv = fn_service(|()| ok::<_, ()>("srv"))
.on_shutdown(|| {
*shutdown.borrow_mut() = true;
})
.clone();
let res = srv.call(()).await; let res = srv.call(()).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!(res.is_ok()); assert!(res.is_ok());
assert_eq!(res.unwrap(), "srv"); assert_eq!(res.unwrap(), "srv");
assert_eq!(
lazy(|cx| srv.poll_shutdown(cx, false)).await,
Poll::Ready(())
);
assert!(*shutdown.borrow());
} }
#[ntex::test] #[ntex::test]