Added example of a concurrent server

Simple example that demonstrates how to create a concurrent
server on top of the primitives exposed by hyper.
This commit is contained in:
Jonathan Reem
2014-09-08 18:15:49 -07:00
parent 77be77a6c1
commit ce61781cbd

View File

@@ -0,0 +1,72 @@
#![feature(macro_rules)]
extern crate hyper;
extern crate debug;
use std::io::util::copy;
use std::io::net::ip::Ipv4Addr;
use std::sync::Arc;
use hyper::{Get, Post};
use hyper::server::{Server, Handler, Incoming, Request, Response};
use hyper::header::common::ContentLength;
trait ConcurrentHandler: Send + Sync {
fn handle(&self, req: Request, res: Response);
}
struct Concurrent<H: ConcurrentHandler> { handler: Arc<H> }
impl<H: ConcurrentHandler> Handler for Concurrent<H> {
fn handle(self, mut incoming: Incoming) {
for (mut req, mut res) in incoming {
let clone = self.handler.clone();
spawn(proc() { clone.handle(req, res) })
}
}
}
macro_rules! try_abort(
($e:expr) => {{
match $e {
Ok(v) => v,
Err(..) => return
}
}}
)
struct Echo;
impl ConcurrentHandler for Echo {
fn handle(&self, mut req: Request, mut res: Response) {
match req.uri {
hyper::uri::AbsolutePath(ref path) => match (&req.method, path.as_slice()) {
(&Get, "/") | (&Get, "/echo") => {
let out = b"Try POST /echo";
res.headers.set(ContentLength(out.len()));
try_abort!(res.write(out));
try_abort!(res.end());
return;
},
(&Post, "/echo") => (), // fall through, fighting mutable borrows
_ => {
res.status = hyper::status::NotFound;
try_abort!(res.end());
return;
}
},
_ => {
try_abort!(res.end());
return;
}
}
try_abort!(copy(&mut req, &mut res));
try_abort!(res.end());
}
}
fn main() {
let server = Server::http(Ipv4Addr(127, 0, 0, 1), 3000);
server.listen(Concurrent { handler: Arc::new(Echo) }).unwrap();
}