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
use hyper::server::{Handler, Server, Request as HyperRequest, Response as HyperResponse};
use hyper::status::StatusCode;
use rustc_serialize::json::{self, Json};
use std::str;
use std::io::Read;
use datatype::{RpcRequest, RpcOk, RpcErr, SocketAddr, Url};
use super::services::Services;
pub struct Edge {
rvi_edge: SocketAddr,
services: Services,
}
impl Edge {
pub fn new(mut services: Services, rvi_edge: SocketAddr, rvi_client: Url) -> Self {
services.register_services(|service| {
let req = RpcRequest::new("register_service", RegisterServiceRequest {
network_address: format!("http://{}", rvi_edge),
service: service.to_string(),
});
let resp = req.send(rvi_client.clone())
.unwrap_or_else(|err| panic!("RegisterServiceRequest failed: {}", err));
let rpc_ok = json::decode::<RpcOk<RegisterServiceResponse>>(&resp)
.unwrap_or_else(|err| panic!("couldn't decode RegisterServiceResponse: {}", err));
rpc_ok.result.expect("expected rpc_ok result").service
});
Edge { rvi_edge: rvi_edge, services: services }
}
pub fn start(&mut self) {
let server = Server::http(&*self.rvi_edge)
.unwrap_or_else(|err| panic!("couldn't start rvi edge server: {}", err));
let _ = server.handle(EdgeHandler::new(self.services.clone())).unwrap();
info!("RVI server edge listening at http://{}.", self.rvi_edge);
}
}
#[derive(RustcEncodable)]
struct RegisterServiceRequest {
pub network_address: String,
pub service: String,
}
#[derive(RustcDecodable)]
struct RegisterServiceResponse {
pub service: String,
pub status: i32,
}
struct EdgeHandler {
services: Services,
}
impl EdgeHandler {
fn new(services: Services) -> EdgeHandler {
EdgeHandler { services: services }
}
}
impl Handler for EdgeHandler {
fn handle(&self, mut req: HyperRequest, mut resp: HyperResponse) {
let mut buf = Vec::new();
req.read_to_end(&mut buf).expect("couldn't read Edge HTTP request body");
let outcome = || -> Result<RpcOk<i32>, RpcErr> {
let text = try!(str::from_utf8(&buf).map_err(|err| RpcErr::parse_error(err.to_string())));
let data = try!(Json::from_str(text).map_err(|err| RpcErr::parse_error(err.to_string())));
let object = try!(data.as_object().ok_or(RpcErr::parse_error("not an object".to_string())));
let id = try!(object.get("id").and_then(|x| x.as_u64())
.ok_or(RpcErr::parse_error("expected id".to_string())));
let method = try!(object.get("method").and_then(|x| x.as_string())
.ok_or(RpcErr::invalid_request(id, "expected method".to_string())));
match method {
"services_available" => Ok(RpcOk::new(id, None)),
"message" => {
let params = try!(object.get("params").and_then(|p| p.as_object())
.ok_or(RpcErr::invalid_request(id, "expected params".to_string())));
let service = try!(params.get("service_name").and_then(|s| s.as_string())
.ok_or(RpcErr::invalid_request(id, "expected params.service_name".to_string())));
self.services.handle_service(service, id, text)
}
_ => Err(RpcErr::method_not_found(id, format!("unknown method: {}", method)))
}
}();
let body = match outcome {
Ok(msg) => {
*resp.status_mut() = StatusCode::Ok;
json::encode::<RpcOk<i32>>(&msg).expect("couldn't encode RpcOk response")
}
Err(err) => {
*resp.status_mut() = StatusCode::BadRequest;
json::encode::<RpcErr>(&err).expect("couldn't encode RpcErr response")
}
};
resp.send(&body.into_bytes()).expect("couldn't send Edge HTTP response");
}
}