Service

Trait Service 

Source
pub trait Service {
    type Request;
    type Response;
    type Error;
    type Future: Future<Output = Result<Self::Response, Self::Error>>;

    // Required methods
    fn poll_ready(
        &mut self,
        ctx: &mut Context<'_>,
    ) -> Poll<Result<(), Self::Error>>;
    fn call(&mut self, req: Self::Request) -> Self::Future;

    // Provided methods
    fn map<F, R>(self, f: F) -> Map<Self, F, R>
       where Self: Sized,
             F: FnMut(Self::Response) -> R { ... }
    fn map_err<F, E>(self, f: F) -> MapErr<Self, F, E>
       where Self: Sized,
             F: Fn(Self::Error) -> E { ... }
}
Expand description

An asynchronous operation from Request to a Response.

The Service trait models a request/response interaction, receiving requests and returning replies. You can think about a service as a function with one argument that returns some result asynchronously. Conceptually, the operation looks like this:

async fn(Request) -> Result<Response, Err>

The Service trait just generalizes this form where each parameter is described as an associated type on the trait. Services can also have mutable state that influence computation.

Service provides a symmetric and uniform API; the same abstractions can be used to represent both clients and servers. Services describe only transformation operations which encourage simple API surfaces. This leads to simpler design of each service, improves test-ability and makes composition easier.

struct MyService;

impl Service for MyService {
     type Request = u8;
     type Response = u64;
     type Error = MyError;
     type Future = Pin<Box<Future<Output=Result<Self::Response, Self::Error>>>>;

     fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> { ... }

     fn call(&mut self, req: Self::Request) -> Self::Future { ... }
}

Sometimes it is not necessary to implement the Service trait. For example, the above service could be rewritten as a simple function and passed to fn_service.

async fn my_service(req: u8) -> Result<u64, MyError>;

Required Associated Types§

Source

type Request

Requests handled by the service.

Source

type Response

Responses given by the service.

Source

type Error

Errors produced by the service.

Source

type Future: Future<Output = Result<Self::Response, Self::Error>>

The future response value.

Required Methods§

Source

fn poll_ready(&mut self, ctx: &mut Context<'_>) -> Poll<Result<(), Self::Error>>

Returns Ready when the service is able to process requests.

If the service is at capacity, then Pending is returned and the task is notified when the service becomes ready again. This function is expected to be called while on a task.

This is a best effort implementation. False positives are permitted. It is permitted for the service to return Ready from a poll_ready call and the next invocation of call results in an error.

§Notes
  1. .poll_ready() might be called on different task from actual service call.
  2. In case of chained services, .poll_ready() get called for all services at once.
Source

fn call(&mut self, req: Self::Request) -> Self::Future

Process the request and return the response asynchronously.

This function is expected to be callable off task. As such, implementations should take care to not call poll_ready. If the service is at capacity and the request is unable to be handled, the returned Future should resolve to an error.

Calling call without calling poll_ready is permitted. The implementation must be resilient to this fact.

Provided Methods§

Source

fn map<F, R>(self, f: F) -> Map<Self, F, R>
where Self: Sized, F: FnMut(Self::Response) -> R,

Map this service’s output to a different type, returning a new service of the resulting type.

This function is similar to the Option::map or Iterator::map where it will change the type of the underlying service.

Note that this function consumes the receiving service and returns a wrapped version of it, similar to the existing map methods in the standard library.

Source

fn map_err<F, E>(self, f: F) -> MapErr<Self, F, E>
where Self: Sized, F: Fn(Self::Error) -> E,

Map this service’s error to a different error, returning a new service.

This function is similar to the Result::map_err where it will change the error type of the underlying service. For example, this can be useful to ensure that services have the same error type.

Note that this function consumes the receiving service and returns a wrapped version of it.

Implementations on Foreign Types§

Source§

impl Service for ExpectHandler

Source§

impl Service for LowResTimeService

Source§

impl<'a, S> Service for &'a mut S
where S: Service + 'a,

Source§

type Request = <S as Service>::Request

Source§

type Response = <S as Service>::Response

Source§

type Error = <S as Service>::Error

Source§

type Future = <S as Service>::Future

Source§

fn poll_ready( &mut self, ctx: &mut Context<'_>, ) -> Poll<Result<(), <&'a mut S as Service>::Error>>

Source§

fn call( &mut self, request: <&'a mut S as Service>::Request, ) -> <S as Service>::Future

Source§

impl<A, B> Service for EitherService<A, B>
where A: Service, B: Service<Response = <A as Service>::Response, Error = <A as Service>::Error>,

Source§

type Request = Either<<A as Service>::Request, <B as Service>::Request>

Source§

type Response = <A as Service>::Response

Source§

type Error = <A as Service>::Error

Source§

type Future = Either<<A as Service>::Future, <B as Service>::Future>

Source§

fn poll_ready( &mut self, cx: &mut Context<'_>, ) -> Poll<Result<(), <EitherService<A, B> as Service>::Error>>

Source§

fn call( &mut self, req: Either<<A as Service>::Request, <B as Service>::Request>, ) -> <EitherService<A, B> as Service>::Future

Source§

impl<R, E, F> Service for KeepAliveService<R, E, F>
where F: Fn() -> E,

Source§

type Request = R

Source§

type Response = R

Source§

type Error = E

Source§

type Future = Ready<Result<R, E>>

Source§

fn poll_ready( &mut self, cx: &mut Context<'_>, ) -> Poll<Result<(), <KeepAliveService<R, E, F> as Service>::Error>>

Source§

fn call(&mut self, req: R) -> <KeepAliveService<R, E, F> as Service>::Future

Source§

impl<S> Service for Box<S>
where S: Service + ?Sized,

Source§

type Request = <S as Service>::Request

Source§

type Response = <S as Service>::Response

Source§

type Error = <S as Service>::Error

Source§

type Future = <S as Service>::Future

Source§

fn poll_ready( &mut self, ctx: &mut Context<'_>, ) -> Poll<Result<(), <S as Service>::Error>>

Source§

fn call( &mut self, request: <Box<S> as Service>::Request, ) -> <S as Service>::Future

Source§

impl<S> Service for Rc<RefCell<S>>
where S: Service,

Source§

type Request = <S as Service>::Request

Source§

type Response = <S as Service>::Response

Source§

type Error = <S as Service>::Error

Source§

type Future = <S as Service>::Future

Source§

fn poll_ready( &mut self, ctx: &mut Context<'_>, ) -> Poll<Result<(), <Rc<RefCell<S>> as Service>::Error>>

Source§

fn call( &mut self, request: <Rc<RefCell<S>> as Service>::Request, ) -> <S as Service>::Future

Source§

impl<S> Service for RefCell<S>
where S: Service,

Source§

type Request = <S as Service>::Request

Source§

type Response = <S as Service>::Response

Source§

type Error = <S as Service>::Error

Source§

type Future = <S as Service>::Future

Source§

fn poll_ready( &mut self, ctx: &mut Context<'_>, ) -> Poll<Result<(), <RefCell<S> as Service>::Error>>

Source§

fn call( &mut self, request: <RefCell<S> as Service>::Request, ) -> <S as Service>::Future

Source§

impl<S> Service for InOrderService<S>
where S: Service, <S as Service>::Response: 'static, <S as Service>::Future: 'static, <S as Service>::Error: 'static,

Source§

type Request = <S as Service>::Request

Source§

type Response = <S as Service>::Response

Source§

type Error = InOrderError<<S as Service>::Error>

Source§

type Future = InOrderServiceResponse<S>

Source§

fn poll_ready( &mut self, cx: &mut Context<'_>, ) -> Poll<Result<(), <InOrderService<S> as Service>::Error>>

Source§

fn call( &mut self, request: <S as Service>::Request, ) -> <InOrderService<S> as Service>::Future

Source§

impl<S> Service for TimeoutService<S>
where S: Service,

Source§

type Request = <S as Service>::Request

Source§

type Response = <S as Service>::Response

Source§

type Error = TimeoutError<<S as Service>::Error>

Source§

type Future = TimeoutServiceResponse<S>

Source§

fn poll_ready( &mut self, cx: &mut Context<'_>, ) -> Poll<Result<(), <TimeoutService<S> as Service>::Error>>

Source§

fn call( &mut self, request: <S as Service>::Request, ) -> <TimeoutService<S> as Service>::Future

Source§

impl<T> Service for TcpConnector<T>
where T: Address,

Source§

type Request = Connect<T>

Source§

type Response = Connection<T, TcpStream>

Source§

type Error = ConnectError

Source§

type Future = Either<TcpConnectorResponse<T>, Ready<Result<<TcpConnector<T> as Service>::Response, <TcpConnector<T> as Service>::Error>>>

Source§

fn poll_ready( &mut self, _: &mut Context<'_>, ) -> Poll<Result<(), <TcpConnector<T> as Service>::Error>>

Source§

fn call(&mut self, req: Connect<T>) -> <TcpConnector<T> as Service>::Future

Source§

impl<T> Service for Resolver<T>
where T: Address,

Source§

type Request = Connect<T>

Source§

type Response = Connect<T>

Source§

type Error = ConnectError

Source§

type Future = Either<Pin<Box<dyn Future<Output = Result<<Resolver<T> as Service>::Response, <Resolver<T> as Service>::Error>>>>, Ready<Result<Connect<T>, <Resolver<T> as Service>::Error>>>

Source§

fn poll_ready( &mut self, _: &mut Context<'_>, ) -> Poll<Result<(), <Resolver<T> as Service>::Error>>

Source§

fn call(&mut self, req: Connect<T>) -> <Resolver<T> as Service>::Future

Source§

impl<T> Service for ConnectService<T>
where T: Address,

Source§

type Request = Connect<T>

Source§

type Response = Connection<T, TcpStream>

Source§

type Error = ConnectError

Source§

type Future = ConnectServiceResponse<T>

Source§

fn poll_ready( &mut self, _: &mut Context<'_>, ) -> Poll<Result<(), <ConnectService<T> as Service>::Error>>

Source§

fn call(&mut self, req: Connect<T>) -> <ConnectService<T> as Service>::Future

Source§

impl<T> Service for TcpConnectService<T>
where T: Address + 'static,

Source§

type Request = Connect<T>

Source§

type Response = TcpStream

Source§

type Error = ConnectError

Source§

type Future = TcpConnectServiceResponse<T>

Source§

fn poll_ready( &mut self, _: &mut Context<'_>, ) -> Poll<Result<(), <TcpConnectService<T> as Service>::Error>>

Source§

fn call(&mut self, req: Connect<T>) -> <TcpConnectService<T> as Service>::Future

Source§

impl<T> Service for UpgradeHandler<T>

Source§

impl<T> Service for InFlightService<T>
where T: Service,

Source§

type Request = <T as Service>::Request

Source§

type Response = <T as Service>::Response

Source§

type Error = <T as Service>::Error

Source§

type Future = InFlightServiceResponse<T>

Source§

fn poll_ready( &mut self, cx: &mut Context<'_>, ) -> Poll<Result<(), <InFlightService<T> as Service>::Error>>

Source§

fn call( &mut self, req: <T as Service>::Request, ) -> <InFlightService<T> as Service>::Future

Source§

impl<T, S, B, X, U> Service for H1ServiceHandler<T, S, B, X, U>
where T: AsyncRead + AsyncWrite + Unpin, S: Service<Request = Request>, <S as Service>::Error: Into<Error>, <S as Service>::Response: Into<Response<B>>, B: MessageBody, X: Service<Request = Request, Response = Request>, <X as Service>::Error: Into<Error>, U: Service<Request = (Request, Framed<T, Codec>), Response = ()>, <U as Service>::Error: Display + Into<Error>,

Source§

type Request = (T, Option<SocketAddr>)

Source§

type Response = ()

Source§

type Error = DispatchError

Source§

type Future = Dispatcher<T, S, B, X, U>

Source§

fn poll_ready( &mut self, cx: &mut Context<'_>, ) -> Poll<Result<(), <H1ServiceHandler<T, S, B, X, U> as Service>::Error>>

Source§

fn call( &mut self, _: <H1ServiceHandler<T, S, B, X, U> as Service>::Request, ) -> <H1ServiceHandler<T, S, B, X, U> as Service>::Future

Implementors§

Source§

impl<A, F, E> Service for MapErr<A, F, E>
where A: Service, F: Fn(<A as Service>::Error) -> E + Clone,

Source§

type Request = <A as Service>::Request

Source§

type Response = <A as Service>::Response

Source§

type Error = E

Source§

type Future = MapErrFuture<A, F, E>

Source§

impl<A, F, Response> Service for Map<A, F, Response>
where A: Service, F: FnMut(<A as Service>::Response) -> Response + Clone,

Source§

type Request = <A as Service>::Request

Source§

type Response = Response

Source§

type Error = <A as Service>::Error

Source§

type Future = MapFuture<A, F, Response>

Source§

impl<F, Fut, Req, Res, Err> Service for FnService<F, Fut, Req, Res, Err>
where F: FnMut(Req) -> Fut, Fut: Future<Output = Result<Res, Err>>,

Source§

type Request = Req

Source§

type Response = Res

Source§

type Error = Err

Source§

type Future = Fut

Source§

impl<F, Fut, Req, Res, Err> Service for FnServiceFactory<F, Fut, Req, Res, Err, ()>
where F: FnMut(Req) -> Fut + Clone, Fut: Future<Output = Result<Res, Err>>,

Source§

type Request = Req

Source§

type Response = Res

Source§

type Error = Err

Source§

type Future = Fut

Source§

impl<T> Service for Pipeline<T>
where T: Service,

Source§

impl<T, F, R, In, Out, Err> Service for Apply<T, F, R, In, Out, Err>
where T: Service<Error = Err>, F: FnMut(In, &mut T) -> R, R: Future<Output = Result<Out, Err>>,

Source§

type Request = In

Source§

type Response = Out

Source§

type Error = Err

Source§

type Future = R