feat(server): make Http compatible with TcpServer
This implements `From<Message> for Request` and `Into<Message> for Response`, allowing an `Http` instance to be used with a `TcpServer` from tokio-proto. Closes #1036 BREAKING CHANGE: This makes `Request.remote_addr` an `Option<SocketAddr>`, instead of `SocketAddr`.
This commit is contained in:
		| @@ -265,18 +265,30 @@ impl<T: Io + 'static, B> Future for __ProtoBindTransport<T, B> { | ||||
|     } | ||||
| } | ||||
|  | ||||
| struct HttpService<T> { | ||||
|     inner: T, | ||||
|     remote_addr: SocketAddr, | ||||
| impl From<Message<__ProtoRequest, http::TokioBody>> for Request { | ||||
|     fn from(message: Message<__ProtoRequest, http::TokioBody>) -> Request { | ||||
|         let (head, body) = match message { | ||||
|             Message::WithoutBody(head) => (head.0, http::Body::empty()), | ||||
|             Message::WithBody(head, body) => (head.0, body.into()), | ||||
|         }; | ||||
|         request::new(None, head, body) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn map_response_to_message<B>(res: Response<B>) -> Message<__ProtoResponse, B> { | ||||
|     let (head, body) = response::split(res); | ||||
| impl<B> Into<Message<__ProtoResponse, B>> for Response<B> { | ||||
|     fn into(self) -> Message<__ProtoResponse, B> { | ||||
|         let (head, body) = response::split(self); | ||||
|         if let Some(body) = body { | ||||
|             Message::WithBody(__ProtoResponse(head), body.into()) | ||||
|         } else { | ||||
|             Message::WithoutBody(__ProtoResponse(head)) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| struct HttpService<T> { | ||||
|     inner: T, | ||||
|     remote_addr: SocketAddr, | ||||
| } | ||||
|  | ||||
| type ResponseHead = http::MessageHead<::StatusCode>; | ||||
| @@ -296,8 +308,8 @@ impl<T, B> Service for HttpService<T> | ||||
|             Message::WithoutBody(head) => (head.0, http::Body::empty()), | ||||
|             Message::WithBody(head, body) => (head.0, body.into()), | ||||
|         }; | ||||
|         let req = request::new(self.remote_addr, head, body); | ||||
|         self.inner.call(req).map(map_response_to_message) | ||||
|         let req = request::new(Some(self.remote_addr), head, body); | ||||
|         self.inner.call(req).map(Into::into) | ||||
|     } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -18,7 +18,7 @@ pub struct Request { | ||||
|     uri: Uri, | ||||
|     version: HttpVersion, | ||||
|     headers: Headers, | ||||
|     remote_addr: SocketAddr, | ||||
|     remote_addr: Option<SocketAddr>, | ||||
|     body: Body, | ||||
| } | ||||
|  | ||||
| @@ -40,8 +40,11 @@ impl Request { | ||||
|     pub fn version(&self) -> &HttpVersion { &self.version } | ||||
|  | ||||
|     /// The remote socket address of this request | ||||
|     /// | ||||
|     /// This is an `Option`, because some underlying transports may not have | ||||
|     /// a socket address, such as Unix Sockets. | ||||
|     #[inline] | ||||
|     pub fn remote_addr(&self) -> &SocketAddr { &self.remote_addr } | ||||
|     pub fn remote_addr(&self) -> Option<&SocketAddr> { self.remote_addr.as_ref() } | ||||
|  | ||||
|     /// The target path of this Request. | ||||
|     #[inline] | ||||
| @@ -82,9 +85,9 @@ impl fmt::Debug for Request { | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn new(addr: SocketAddr, incoming: RequestHead, body: Body) -> Request { | ||||
| pub fn new(addr: Option<SocketAddr>, incoming: RequestHead, body: Body) -> Request { | ||||
|     let MessageHead { version, subject: RequestLine(method, uri), headers } = incoming; | ||||
|     debug!("Request::new: addr={}, req=\"{} {} {}\"", addr, method, uri, version); | ||||
|     debug!("Request::new: addr={}, req=\"{} {} {}\"", MaybeAddr(&addr), method, uri, version); | ||||
|     debug!("Request::new: headers={:?}", headers); | ||||
|  | ||||
|     Request { | ||||
| @@ -96,3 +99,15 @@ pub fn new(addr: SocketAddr, incoming: RequestHead, body: Body) -> Request { | ||||
|         body: body, | ||||
|     } | ||||
| } | ||||
|  | ||||
| struct MaybeAddr<'a>(&'a Option<SocketAddr>); | ||||
|  | ||||
| impl<'a> fmt::Display for MaybeAddr<'a> { | ||||
|     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||
|         match *self.0 { | ||||
|             Some(ref addr) => fmt::Display::fmt(addr, f), | ||||
|             None => f.write_str("None"), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user