1 | use futures_core::TryFuture; |
2 | use futures_util::{future, TryFutureExt}; |
3 | use std::fmt; |
4 | use std::future::Future; |
5 | use std::pin::Pin; |
6 | use std::task::{Context, Poll}; |
7 | use tower_layer::Layer; |
8 | use tower_service::Service; |
9 | |
10 | /// Service returned by the [`and_then`] combinator. |
11 | /// |
12 | /// [`and_then`]: crate::util::ServiceExt::and_then |
13 | #[derive (Clone)] |
14 | pub struct AndThen<S, F> { |
15 | inner: S, |
16 | f: F, |
17 | } |
18 | |
19 | impl<S, F> fmt::Debug for AndThen<S, F> |
20 | where |
21 | S: fmt::Debug, |
22 | { |
23 | fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { |
24 | f&mut DebugStruct<'_, '_>.debug_struct("AndThen" ) |
25 | .field("inner" , &self.inner) |
26 | .field(name:"f" , &format_args!(" {}" , std::any::type_name::<F>())) |
27 | .finish() |
28 | } |
29 | } |
30 | |
31 | pin_project_lite::pin_project! { |
32 | /// Response future from [`AndThen`] services. |
33 | /// |
34 | /// [`AndThen`]: crate::util::AndThen |
35 | pub struct AndThenFuture<F1, F2: TryFuture, N> { |
36 | #[pin] |
37 | inner: future::AndThen<future::ErrInto<F1, F2::Error>, F2, N>, |
38 | } |
39 | } |
40 | |
41 | impl<F1, F2: TryFuture, N> AndThenFuture<F1, F2, N> { |
42 | pub(crate) fn new(inner: future::AndThen<future::ErrInto<F1, F2::Error>, F2, N>) -> Self { |
43 | Self { inner } |
44 | } |
45 | } |
46 | |
47 | impl<F1, F2: TryFuture, N> std::fmt::Debug for AndThenFuture<F1, F2, N> { |
48 | fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { |
49 | f&mut DebugTuple<'_, '_>.debug_tuple(name:"AndThenFuture" ) |
50 | .field(&format_args!("..." )) |
51 | .finish() |
52 | } |
53 | } |
54 | |
55 | impl<F1, F2: TryFuture, N> Future for AndThenFuture<F1, F2, N> |
56 | where |
57 | future::AndThen<future::ErrInto<F1, F2::Error>, F2, N>: Future, |
58 | { |
59 | type Output = <future::AndThen<future::ErrInto<F1, F2::Error>, F2, N> as Future>::Output; |
60 | |
61 | #[inline ] |
62 | fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> { |
63 | self.project().inner.poll(cx) |
64 | } |
65 | } |
66 | |
67 | /// A [`Layer`] that produces a [`AndThen`] service. |
68 | /// |
69 | /// [`Layer`]: tower_layer::Layer |
70 | #[derive (Clone, Debug)] |
71 | pub struct AndThenLayer<F> { |
72 | f: F, |
73 | } |
74 | |
75 | impl<S, F> AndThen<S, F> { |
76 | /// Creates a new `AndThen` service. |
77 | pub const fn new(inner: S, f: F) -> Self { |
78 | AndThen { f, inner } |
79 | } |
80 | |
81 | /// Returns a new [`Layer`] that produces [`AndThen`] services. |
82 | /// |
83 | /// This is a convenience function that simply calls [`AndThenLayer::new`]. |
84 | /// |
85 | /// [`Layer`]: tower_layer::Layer |
86 | pub fn layer(f: F) -> AndThenLayer<F> { |
87 | AndThenLayer { f } |
88 | } |
89 | } |
90 | |
91 | impl<S, F, Request, Fut> Service<Request> for AndThen<S, F> |
92 | where |
93 | S: Service<Request>, |
94 | S::Error: Into<Fut::Error>, |
95 | F: FnOnce(S::Response) -> Fut + Clone, |
96 | Fut: TryFuture, |
97 | { |
98 | type Response = Fut::Ok; |
99 | type Error = Fut::Error; |
100 | type Future = AndThenFuture<S::Future, Fut, F>; |
101 | |
102 | fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> { |
103 | self.inner.poll_ready(cx).map_err(Into::into) |
104 | } |
105 | |
106 | fn call(&mut self, request: Request) -> Self::Future { |
107 | AndThenFuture::new(self.inner.call(req:request).err_into().and_then(self.f.clone())) |
108 | } |
109 | } |
110 | |
111 | impl<F> AndThenLayer<F> { |
112 | /// Creates a new [`AndThenLayer`] layer. |
113 | pub const fn new(f: F) -> Self { |
114 | AndThenLayer { f } |
115 | } |
116 | } |
117 | |
118 | impl<S, F> Layer<S> for AndThenLayer<F> |
119 | where |
120 | F: Clone, |
121 | { |
122 | type Service = AndThen<S, F>; |
123 | |
124 | fn layer(&self, inner: S) -> Self::Service { |
125 | AndThen { |
126 | f: self.f.clone(), |
127 | inner, |
128 | } |
129 | } |
130 | } |
131 | |