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
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
use std::net::SocketAddr;
use std::panic::AssertUnwindSafe;
use std::pin::Pin;
use std::sync::Arc;
use futures::prelude::*;
use futures::task::{self, Poll};
use hyper::service::Service;
use hyper::{Body, Request, Response};
use crate::handler::NewHandler;
use crate::state::State;
mod trap;
pub use trap::call_handler;
pub(crate) struct GothamService<T>
where
T: NewHandler + 'static,
{
handler: Arc<T>,
}
impl<T> GothamService<T>
where
T: NewHandler + 'static,
{
pub(crate) fn new(handler: T) -> GothamService<T> {
GothamService {
handler: Arc::new(handler),
}
}
pub(crate) fn connect(&self, client_addr: SocketAddr) -> ConnectedGothamService<T> {
ConnectedGothamService {
client_addr,
handler: self.handler.clone(),
}
}
}
pub(crate) struct ConnectedGothamService<T>
where
T: NewHandler + 'static,
{
handler: Arc<T>,
client_addr: SocketAddr,
}
impl<T> Service<Request<Body>> for ConnectedGothamService<T>
where
T: NewHandler,
{
type Response = Response<Body>;
type Error = anyhow::Error;
type Future = Pin<Box<dyn Future<Output = Result<Self::Response, Self::Error>> + Send>>;
fn poll_ready(
&mut self,
_cx: &mut task::Context<'_>,
) -> Poll<std::result::Result<(), Self::Error>> {
Poll::Ready(Ok(()))
}
fn call<'a>(&'a mut self, req: Request<Body>) -> Self::Future {
let state = State::from_request(req, self.client_addr);
call_handler(self.handler.clone(), AssertUnwindSafe(state)).boxed()
}
}
#[cfg(test)]
mod tests {
use super::*;
use hyper::{Body, StatusCode};
use crate::helpers::http::response::create_empty_response;
use crate::router::builder::*;
use crate::state::State;
fn handler(state: State) -> (State, Response<Body>) {
let res = create_empty_response(&state, StatusCode::ACCEPTED);
(state, res)
}
#[test]
fn new_handler_closure() {
let service = GothamService::new(|| Ok(handler));
let req = Request::get("http://localhost/")
.body(Body::empty())
.unwrap();
let f = service
.connect("127.0.0.1:10000".parse().unwrap())
.call(req);
let response = futures::executor::block_on(f).unwrap();
assert_eq!(response.status(), StatusCode::ACCEPTED);
}
#[test]
fn router() {
let router = build_simple_router(|route| {
route.get("/").to(handler);
});
let service = GothamService::new(router);
let req = Request::get("http://localhost/")
.body(Body::empty())
.unwrap();
let f = service
.connect("127.0.0.1:10000".parse().unwrap())
.call(req);
let response = futures::executor::block_on(f).unwrap();
assert_eq!(response.status(), StatusCode::ACCEPTED);
}
}