1
0
mirror of https://github.com/fafhrd91/actix-web synced 2025-03-14 20:26:26 +01:00
actix-web/src/service/apply.rs

226 lines
5.7 KiB
Rust
Raw Normal View History

use std::marker::PhantomData;
2018-08-31 12:51:26 -07:00
use futures::{Async, Future, IntoFuture, Poll};
2018-09-11 09:30:22 -07:00
2018-09-17 16:16:42 -07:00
use super::{IntoNewService, IntoService, NewService, Service};
2018-08-31 12:51:26 -07:00
/// `Apply` service combinator
2018-11-29 16:56:15 -10:00
pub struct Apply<T, F, In, Out, Request>
where
T: Service<Request>,
{
service: T,
f: F,
2018-11-29 16:56:15 -10:00
r: PhantomData<(In, Out, Request)>,
}
2018-11-29 16:56:15 -10:00
impl<T, F, In, Out, Request> Apply<T, F, In, Out, Request>
where
2018-11-29 16:56:15 -10:00
T: Service<Request>,
F: Fn(In, &mut T) -> Out,
Out: IntoFuture,
{
/// Create new `Apply` combinator
2018-11-29 16:56:15 -10:00
pub fn new<I: IntoService<T, Request>>(service: I, f: F) -> Self {
Self {
2018-09-17 16:16:42 -07:00
service: service.into_service(),
f,
r: PhantomData,
}
}
}
2018-11-29 16:56:15 -10:00
impl<T, F, In, Out, Request> Clone for Apply<T, F, In, Out, Request>
where
2018-11-29 16:56:15 -10:00
T: Service<Request> + Clone,
F: Clone,
{
fn clone(&self) -> Self {
Apply {
service: self.service.clone(),
f: self.f.clone(),
r: PhantomData,
}
}
}
2018-11-29 16:56:15 -10:00
impl<T, F, In, Out, Request> Service<In> for Apply<T, F, In, Out, Request>
where
2018-11-29 16:56:15 -10:00
T: Service<Request, Error = Out::Error>,
F: Fn(In, &mut T) -> Out,
Out: IntoFuture,
{
2018-11-29 16:56:15 -10:00
type Response = <Out::Future as Future>::Item;
type Error = <Out::Future as Future>::Error;
type Future = Out::Future;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
self.service.poll_ready().map_err(|e| e.into())
}
2018-11-29 16:56:15 -10:00
fn call(&mut self, req: In) -> Self::Future {
2018-08-31 12:51:26 -07:00
(self.f)(req, &mut self.service).into_future()
}
}
2018-08-30 17:54:59 -07:00
/// `ApplyNewService` new service combinator
2018-11-29 16:56:15 -10:00
pub struct ApplyNewService<T, F, In, Out, Request>
where
T: NewService<Request>,
{
service: T,
f: F,
2018-11-29 16:56:15 -10:00
r: PhantomData<(In, Out, Request)>,
}
2018-11-29 16:56:15 -10:00
impl<T, F, In, Out, Request> ApplyNewService<T, F, In, Out, Request>
where
2018-11-29 16:56:15 -10:00
T: NewService<Request>,
F: Fn(In, &mut T::Service) -> Out,
Out: IntoFuture,
{
2018-08-30 17:54:59 -07:00
/// Create new `ApplyNewService` new service instance
2018-11-29 16:56:15 -10:00
pub fn new<F1: IntoNewService<T, Request>>(service: F1, f: F) -> Self {
Self {
f,
service: service.into_new_service(),
r: PhantomData,
}
}
}
2018-11-29 16:56:15 -10:00
impl<T, F, In, Out, Request> Clone for ApplyNewService<T, F, In, Out, Request>
where
2018-11-29 16:56:15 -10:00
T: NewService<Request> + Clone,
F: Fn(Out, &mut T::Service) -> Out + Clone,
Out: IntoFuture,
{
fn clone(&self) -> Self {
Self {
service: self.service.clone(),
f: self.f.clone(),
r: PhantomData,
}
}
}
2018-11-29 16:56:15 -10:00
impl<T, F, In, Out, Request> NewService<In> for ApplyNewService<T, F, In, Out, Request>
where
2018-11-29 16:56:15 -10:00
T: NewService<Request, Error = Out::Error>,
F: Fn(In, &mut T::Service) -> Out + Clone,
Out: IntoFuture,
{
2018-11-29 16:56:15 -10:00
type Response = <Out::Future as Future>::Item;
type Error = <Out::Future as Future>::Error;
type Service = Apply<T::Service, F, In, Out, Request>;
type InitError = T::InitError;
2018-11-29 16:56:15 -10:00
type Future = ApplyNewServiceFuture<T, F, In, Out, Request>;
fn new_service(&self) -> Self::Future {
2018-08-30 17:54:59 -07:00
ApplyNewServiceFuture::new(self.service.new_service(), self.f.clone())
}
}
2018-11-29 16:56:15 -10:00
pub struct ApplyNewServiceFuture<T, F, In, Out, Request>
where
2018-11-29 16:56:15 -10:00
T: NewService<Request>,
F: Fn(In, &mut T::Service) -> Out,
Out: IntoFuture,
{
fut: T::Future,
f: Option<F>,
2018-11-29 16:56:15 -10:00
r: PhantomData<(In, Out)>,
}
2018-11-29 16:56:15 -10:00
impl<T, F, In, Out, Request> ApplyNewServiceFuture<T, F, In, Out, Request>
where
2018-11-29 16:56:15 -10:00
T: NewService<Request>,
F: Fn(In, &mut T::Service) -> Out,
Out: IntoFuture,
{
fn new(fut: T::Future, f: F) -> Self {
2018-08-30 17:54:59 -07:00
ApplyNewServiceFuture {
f: Some(f),
fut,
r: PhantomData,
}
}
}
2018-11-29 16:56:15 -10:00
impl<T, F, In, Out, Request> Future for ApplyNewServiceFuture<T, F, In, Out, Request>
where
2018-11-29 16:56:15 -10:00
T: NewService<Request>,
F: Fn(In, &mut T::Service) -> Out,
Out: IntoFuture,
{
2018-11-29 16:56:15 -10:00
type Item = Apply<T::Service, F, In, Out, Request>;
type Error = T::InitError;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
if let Async::Ready(service) = self.fut.poll()? {
2018-09-17 15:53:41 -07:00
Ok(Async::Ready(Apply::new(service, self.f.take().unwrap())))
} else {
Ok(Async::NotReady)
}
}
}
2018-09-12 13:34:53 -07:00
#[cfg(test)]
mod tests {
use futures::future::{ok, FutureResult};
use futures::{Async, Future, Poll};
2018-09-17 19:21:24 -07:00
use service::{
IntoNewService, IntoService, NewService, NewServiceExt, Service, ServiceExt,
};
2018-09-12 13:34:53 -07:00
#[derive(Clone)]
struct Srv;
impl Service for Srv {
type Request = ();
type Response = ();
type Error = ();
type Future = FutureResult<(), ()>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
Ok(Async::Ready(()))
}
fn call(&mut self, _: ()) -> Self::Future {
ok(())
}
}
#[test]
fn test_call() {
2018-09-17 16:50:35 -07:00
let blank = |req| Ok(req);
let mut srv = blank.into_service().apply(Srv, |req: &'static str, srv| {
srv.call(()).map(move |res| (req, res))
});
assert!(srv.poll_ready().is_ok());
2018-09-12 13:34:53 -07:00
let res = srv.call("srv").poll();
assert!(res.is_ok());
assert_eq!(res.unwrap(), Async::Ready(("srv", ())));
}
2018-09-17 19:21:24 -07:00
#[test]
fn test_new_service() {
let blank = || Ok::<_, ()>((|req| Ok(req)).into_service());
let new_srv = blank.into_new_service().apply(
|| Ok(Srv),
|req: &'static str, srv| srv.call(()).map(move |res| (req, res)),
);
if let Async::Ready(mut srv) = new_srv.new_service().poll().unwrap() {
assert!(srv.poll_ready().is_ok());
let res = srv.call("srv").poll();
assert!(res.is_ok());
assert_eq!(res.unwrap(), Async::Ready(("srv", ())));
} else {
panic!()
}
}
2018-09-12 13:34:53 -07:00
}