diff --git a/ntex-service/CHANGES.md b/ntex-service/CHANGES.md index be61f60d..aeb789f4 100644 --- a/ntex-service/CHANGES.md +++ b/ntex-service/CHANGES.md @@ -1,5 +1,9 @@ # Changes +## [0.1.6] - 2021-03-26 + +* Add .on_shutdown() callback to fn_service + ## [0.1.5] - 2021-01-13 * Use pin-project-lite instead of pin-project diff --git a/ntex-service/Cargo.toml b/ntex-service/Cargo.toml index 214fab89..767650d7 100644 --- a/ntex-service/Cargo.toml +++ b/ntex-service/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "ntex-service" -version = "0.1.5" +version = "0.1.6" authors = ["ntex contributors "] description = "ntex service" keywords = ["network", "framework", "async", "futures"] diff --git a/ntex-service/src/fn_service.rs b/ntex-service/src/fn_service.rs index e080b475..fb4c7503 100644 --- a/ntex-service/src/fn_service.rs +++ b/ntex-service/src/fn_service.rs @@ -1,4 +1,4 @@ -use std::cell::RefCell; +use std::cell::{Cell, RefCell}; use std::future::Future; use std::marker::PhantomData; use std::task::{Context, Poll}; @@ -119,12 +119,16 @@ where FnServiceConfig::new(f) } -pub struct FnService +#[inline] +pub fn fn_shutdown() {} + +pub struct FnService where F: Fn(Req) -> Fut, Fut: Future>, { f: F, + f_shutdown: Cell>, _t: PhantomData, } @@ -134,24 +138,49 @@ where Fut: Future>, { 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(self, f: FShut) -> FnService + where + FShut: FnOnce(), + { + FnService { + f: self.f, + f_shutdown: Cell::new(Some(f)), + _t: PhantomData, + } } } -impl Clone for FnService +impl Clone for FnService where F: Fn(Req) -> Fut + Clone, + FShut: FnOnce() + Clone, Fut: Future>, { #[inline] 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 Service for FnService +impl Service for FnService where F: Fn(Req) -> Fut, + FShut: FnOnce(), Fut: Future>, { type Request = Req; @@ -164,6 +193,14 @@ where Poll::Ready(Ok(())) } + #[inline] + fn poll_shutdown(&self, _: &mut Context<'_>, _: bool) -> Poll<()> { + if let Some(f) = self.f_shutdown.take() { + (f)() + } + Poll::Ready(()) + } + #[inline] fn call(&self, req: Req) -> Self::Future { (self.f)(req) @@ -181,12 +218,13 @@ where } } -pub struct FnServiceFactory +pub struct FnServiceFactory where F: Fn(Req) -> Fut, Fut: Future>, { f: F, + f_shutdown: Cell>, _t: PhantomData<(Req, Cfg)>, } @@ -196,24 +234,54 @@ where Fut: Future>, { 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( + self, + f: FShut, + ) -> FnServiceFactory + where + FShut: FnOnce(), + { + FnServiceFactory { + f: self.f, + f_shutdown: Cell::new(Some(f)), + _t: PhantomData, + } } } -impl Clone for FnServiceFactory +impl Clone + for FnServiceFactory where F: Fn(Req) -> Fut + Clone, + FShut: FnOnce() + Clone, Fut: Future>, { #[inline] 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 Service for FnServiceFactory +impl Service + for FnServiceFactory where - F: Fn(Req) -> Fut + Clone, + F: Fn(Req) -> Fut, + FShut: FnOnce(), Fut: Future>, { type Request = Req; @@ -226,16 +294,25 @@ where Poll::Ready(Ok(())) } + #[inline] + fn poll_shutdown(&self, _: &mut Context<'_>, _: bool) -> Poll<()> { + if let Some(f) = self.f_shutdown.take() { + (f)() + } + Poll::Ready(()) + } + #[inline] fn call(&self, req: Self::Request) -> Self::Future { (self.f)(req) } } -impl ServiceFactory - for FnServiceFactory +impl ServiceFactory + for FnServiceFactory where F: Fn(Req) -> Fut + Clone, + FShut: FnOnce() + Clone, Fut: Future>, { type Request = Req; @@ -243,13 +320,20 @@ where type Error = Err; type Config = Cfg; - type Service = FnService; + type Service = FnService; type InitError = (); type Future = Ready>; #[inline] 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)] mod tests { - use std::task::Poll; + use std::{rc::Rc, task::Poll}; use futures_util::future::{lazy, ok}; @@ -454,7 +538,12 @@ mod tests { #[ntex::test] 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 res = srv.call(()).await; @@ -466,6 +555,12 @@ mod tests { let res = srv2.call(()).await; assert!(res.is_ok()); assert_eq!(res.unwrap(), "srv"); + + assert_eq!( + lazy(|cx| srv2.poll_shutdown(cx, false)).await, + Poll::Ready(()) + ); + assert!(*shutdown.borrow()); } #[ntex::test] @@ -480,12 +575,22 @@ mod tests { #[ntex::test] 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; assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(()))); assert!(res.is_ok()); assert_eq!(res.unwrap(), "srv"); + assert_eq!( + lazy(|cx| srv.poll_shutdown(cx, false)).await, + Poll::Ready(()) + ); + assert!(*shutdown.borrow()); } #[ntex::test]