use std::net::{TcpListener, TcpStream, Shutdown};
use std::io::Write;
use std::str;
use std::fmt;
use crate::request::Request;
use crate::response::Response;
use crate::http_method::HttpMethod;
pub trait HttpHandler {
fn do_get(&self, _request: Request) -> Response {
return self.default_action(_request);
}
fn do_head(&self, _request: Request) -> Response {
return self.default_action(_request);
}
fn do_post(&self, _request: Request) -> Response {
return self.default_action(_request);
}
fn do_put(&self, _request: Request) -> Response {
return self.default_action(_request);
}
fn do_delete(&self, _request: Request) -> Response {
return self.default_action(_request);
}
fn do_connect(&self, _request: Request) -> Response {
return self.default_action(_request);
}
fn do_options(&self, _request: Request) -> Response {
return self.default_action(_request);
}
fn do_trace(&self, _request: Request) -> Response {
return self.default_action(_request);
}
fn do_patch(&self, _request: Request) -> Response {
return self.default_action(_request);
}
fn default_action(&self, _request: Request) -> Response {
let msg = String::from("
Method not allowed
");
return Response::html(msg, 405);
}
}
pub struct HttpServer {
host: String,
port: u16,
socket: TcpListener,
handler: T
}
impl HttpServer where T: HttpHandler {
// TODO(andrew): add explanations for errors?
pub fn new(host: &str, port: u16, handler: T) -> Result, &str> {
let addr = format!("{}:{}", host, port);
let sock = TcpListener::bind(addr);
match sock {
Ok(s) => {
let server = HttpServer {
host: String::from(host),
port: port,
socket: s,
handler: handler,
};
return Ok(server);
},
Err(_) => return Err("Couldn't start server")
};
}
pub fn serve_forever(&self) -> Result<(), &str> {
for stream in self.socket.incoming() {
match stream {
Ok(s) => {
// TODO(andrew): replace println with logging.
println!("Got connection!");
match self.handle_client(&s) {
Ok(_) => (),
Err(msg) => return Err(msg)
};
match s.shutdown(Shutdown::Both) {
Ok(_) => println!("Closed connection"),
Err(_) => return Err("Couldn't close client socket")
};
},
Err(_) => break
};
}
return Ok(());
}
fn handle_client(&self, mut stream: &TcpStream) -> Result<(), &str> {
let mut buf: [u8; 1024] = [0; 1024];
// TODO(andrew): read all body, not first 1024 bytes.
stream.peek(&mut buf).expect("Couldn't read from socket");
let request = Request::from(&buf);
let request = match request {
Some(r) => r,
None => return Err("Request parsed with errors")
};
let response = match request.method {
HttpMethod::GET => self.handler.do_get(request),
HttpMethod::HEAD => self.handler.do_head(request),
HttpMethod::POST => self.handler.do_post(request),
HttpMethod::PUT => self.handler.do_put(request),
HttpMethod::DELETE => self.handler.do_delete(request),
HttpMethod::CONNECT => self.handler.do_connect(request),
HttpMethod::OPTIONS => self.handler.do_options(request),
HttpMethod::TRACE => self.handler.do_trace(request),
HttpMethod::PATCH => self.handler.do_patch(request),
};
let response = response.format();
match stream.write(&response) {
Ok(_) => return Ok(()),
Err(_) => return Err("Couldn't write to client socket")
};
}
}
impl fmt::Display for HttpServer where T: HttpHandler {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
return write!(f, "HttpServer({}:{})", self.host, self.port);
}
}