1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
use futures::{Future, Poll};
use http;
use tower_service::Service;
use std::time::Instant;
#[derive(Debug)]
pub struct LogService<S> {
inner: S,
target: &'static str,
}
#[derive(Debug)]
pub struct ResponseFuture<T> {
inner: T,
method: http::Method,
path: Option<http::uri::PathAndQuery>,
version: http::Version,
start: Instant,
target: &'static str,
}
impl<S> LogService<S> {
pub(super) fn new(inner: S, target: &'static str) -> LogService<S> {
LogService {
inner,
target,
}
}
}
impl<S, RequestBody, ResponseBody> Service for LogService<S>
where S: Service<Request = http::Request<RequestBody>,
Response = http::Response<ResponseBody>>,
S::Error: ::std::error::Error,
{
type Request = S::Request;
type Response = S::Response;
type Error = S::Error;
type Future = ResponseFuture<S::Future>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
self.inner.poll_ready()
}
fn call(&mut self, request: Self::Request) -> Self::Future {
let method = request.method().clone();
let path = request.uri().path_and_query().map(|p| p.clone());
let version = request.version();
let start = Instant::now();
let inner = self.inner.call(request);
ResponseFuture {
inner,
method,
path,
version,
start,
target: self.target,
}
}
}
impl<T, B> Future for ResponseFuture<T>
where
T: Future<Item = http::Response<B>>,
T::Error: ::std::error::Error,
{
type Item = T::Item;
type Error = T::Error;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
use futures::Async::*;
let res = self.inner.poll();
match res {
Ok(Ready(ref response)) => {
let full_path = self.path.as_ref()
.map(|p| p.as_str())
.unwrap_or("/");
info!(
target: self.target,
"\"{} {} {:?}\" {} {:?}",
self.method,
full_path,
self.version,
response.status().as_u16(),
self.start.elapsed(),
);
}
Err(ref err) => {
warn!("ERROR: {}", err);
}
_ => {}
}
res
}
}