Rocket/lib/src/response/responder.rs

92 lines
3.2 KiB
Rust
Raw Normal View History

use std::io::{Read, Write};
use std::fs::File;
use std::fmt;
use response::ResponseOutcome;
use http::mime::{Mime, TopLevel, SubLevel};
use http::hyper::{header, FreshHyperResponse, StatusCode};
2016-04-02 08:51:22 +00:00
// TODO: Have this return something saying whether every was okay. Need
// something like to be able to forward requests on when things don't work out.
// In particular, we want to try the next ranked route when when parsing
// parameters doesn't work out.
pub trait Responder {
fn respond<'a>(&mut self, mut res: FreshHyperResponse<'a>) -> ResponseOutcome<'a>;
}
impl<'a> Responder for &'a str {
fn respond<'b>(&mut self, mut res: FreshHyperResponse<'b>) -> ResponseOutcome<'b> {
if res.headers().get::<header::ContentType>().is_none() {
let mime = Mime(TopLevel::Text, SubLevel::Plain, vec![]);
res.headers_mut().set(header::ContentType(mime));
}
ResponseOutcome::of(res.send(self.as_bytes()))
}
}
impl Responder for String {
fn respond<'a>(&mut self, mut res: FreshHyperResponse<'a>) -> ResponseOutcome<'a> {
if res.headers().get::<header::ContentType>().is_none() {
let mime = Mime(TopLevel::Text, SubLevel::Html, vec![]);
res.headers_mut().set(header::ContentType(mime));
}
2016-10-09 11:29:02 +00:00
ResponseOutcome::of(res.send(self.as_bytes()))
}
}
impl Responder for File {
fn respond<'a>(&mut self, mut res: FreshHyperResponse<'a>) -> ResponseOutcome<'a> {
2016-10-09 11:29:02 +00:00
let size = match self.metadata() {
Ok(md) => md.len(),
Err(e) => {
error_!("Failed to read file metadata: {:?}", e);
return ResponseOutcome::forward(StatusCode::InternalServerError, res);
2016-10-09 11:29:02 +00:00
}
};
let mut v = Vec::new();
2016-10-09 11:29:02 +00:00
if let Err(e) = self.read_to_end(&mut v) {
error_!("Failed to read file: {:?}", e);
return ResponseOutcome::forward(StatusCode::InternalServerError, res);
2016-10-09 11:29:02 +00:00
}
2016-10-09 11:29:02 +00:00
res.headers_mut().set(header::ContentLength(size));
ResponseOutcome::of(res.start().and_then(|mut stream| stream.write_all(&v)))
}
}
impl<T: Responder> Responder for Option<T> {
fn respond<'a>(&mut self, res: FreshHyperResponse<'a>) -> ResponseOutcome<'a> {
2016-10-09 11:29:02 +00:00
if let Some(ref mut val) = *self {
val.respond(res)
} else {
warn_!("Response was `None`.");
ResponseOutcome::forward(StatusCode::NotFound, res)
}
}
}
impl<T: Responder, E: fmt::Debug> Responder for Result<T, E> {
// prepend with `default` when using impl specialization
default fn respond<'a>(&mut self, res: FreshHyperResponse<'a>) -> ResponseOutcome<'a> {
2016-10-09 11:29:02 +00:00
match *self {
Ok(ref mut val) => val.respond(res),
Err(ref e) => {
error_!("{:?}", e);
ResponseOutcome::forward(StatusCode::InternalServerError, res)
2016-10-09 11:29:02 +00:00
}
}
}
}
impl<T: Responder, E: Responder + fmt::Debug> Responder for Result<T, E> {
fn respond<'a>(&mut self, res: FreshHyperResponse<'a>) -> ResponseOutcome<'a> {
match *self {
Ok(ref mut responder) => responder.respond(res),
Err(ref mut responder) => responder.respond(res),
}
}
}