My application uses the hyper crate to serve some data over HTTP. The core is a handler function, like this:
struct HttpHandler {}
impl hyper::server::Handler for HttpHandler {
fn handle(&self, req: hyper::server::Request, res: hyper::server::Response) {
res.send(b"Hello").unwrap();
}
}
Hyper will call this function for each HTTP request, providing the Request
req
and Response
res
variables.
I want to unit test my handle
function, so I call the function, providing a Request
and Response
, and assert that the Response
has been used to send the expected data ("Hello").
I'm trying to instantiate a Request
and a Response
object, to pass into the handle
function. For this, several dependencies are needed, which I need to create. For this, I ended up implementing a mock NetworkStream
:
mod tests {
use std::io;
use std::io::prelude::*;
use std::net::{IpAddr, Ipv4Addr, SocketAddr, SocketAddrV4};
use std::time::Duration;
use hyper::server::Handler;
use super::*;
struct MockNetworkStream {}
impl Read for MockNetworkStream {
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
Ok(1)
}
}
impl Write for MockNetworkStream {
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
Ok(1)
}
fn flush(&mut self) -> io::Result<()> {
Ok(())
}
}
impl hyper::net::NetworkStream for MockNetworkStream {
fn peer_addr(&mut self) -> Result<SocketAddr, io::Error> {
Ok(SocketAddr::V4(SocketAddrV4::new(Ipv4Addr::new(127, 0, 0, 1), 8080)))
}
fn set_read_timeout(&self, dur: Option<Duration>) -> Result<(), io::Error> {
Ok(())
}
fn set_write_timeout(&self, dur: Option<Duration>) -> Result<(), io::Error> {
Ok(())
}
}
#[test]
fn test_handle() {
let handler = HttpHandler {};
let mut request_mock_network_stream = MockNetworkStream {};
let mut reader = hyper::buffer::BufReader::new(&mut request_mock_network_stream as
&mut hyper::net::NetworkStream);
let socket = SocketAddr::new(IpAddr::V4(Ipv4Addr::new(127, 0, 0, 1)), 8080);
// The following fails with
// 'tests::test_handle' panicked at 'called `Result::unwrap()` on an `Err` value: Header'
let request = hyper::server::Request::new(&mut reader, socket).unwrap();
let mut headers = hyper::header::Headers::new();
let mut response_mock_network_stream = MockNetworkStream {};
let response = hyper::server::Response::new(&mut response_mock_network_stream,
&mut headers);
handler.handle(request, response);
// I would like to do some assert like this:
// assert_eq!(result, b"Hello");
}
}
Full runnable playground example
However, instantiating the Request
panics:
// The following fails with
// 'tests::test_handle' panicked at 'called `Result::unwrap()` on an `Err` value: Header'
let request = hyper::server::Request::new(&mut reader, socket).unwrap();
Where is the mistake in my mock setup? Is there a more straightforward way to test such a handler function without so much boilerplate code?