]> piware.de Git - learn-rust.git/commitdiff
async-http: Unit test for handle_connection()
authorMartin Pitt <martin@piware.de>
Fri, 16 Sep 2022 07:39:05 +0000 (09:39 +0200)
committerMartin Pitt <martin@piware.de>
Fri, 16 Sep 2022 07:39:05 +0000 (09:39 +0200)
async-http/src/main.rs

index c2884082c095f96dcb4598483b484ca66e4d92fd..1a8dddb2ed8f7627dece48d51e14d282218ac845 100644 (file)
@@ -2,7 +2,8 @@ use std::fs;
 use std::time::Duration;
 
 use async_std::prelude::*;
-use async_std::net::{ TcpListener, TcpStream };
+use async_std::io::{ Read, Write };
+use async_std::net::{ TcpListener };
 use async_std::task;
 use futures::stream::StreamExt;
 
@@ -19,7 +20,7 @@ async fn main() {
     }).await;
 }
 
-async fn handle_connection(mut stream: TcpStream) {
+async fn handle_connection(mut stream: impl Read + Write + Unpin) {
     // Read the first 1024 bytes of data from the stream
     let mut buffer = [0; 1024];
     assert!(stream.read(&mut buffer).await.unwrap() > 0);
@@ -45,3 +46,70 @@ async fn handle_connection(mut stream: TcpStream) {
     stream.write_all(response.as_bytes()).await.unwrap();
     stream.flush().await.unwrap();
 }
+
+#[cfg(test)]
+mod tests {
+    use super::*;
+
+    use std::cmp;
+    use std::pin::Pin;
+
+    use futures::io::Error;
+    use futures::task::{Context, Poll};
+
+    struct MockTcpStream {
+        read_data: Vec<u8>,
+        write_data: Vec<u8>,
+    }
+
+    impl Read for MockTcpStream {
+        fn poll_read(
+            self: Pin<&mut Self>,
+            _: &mut Context,
+            buf: &mut [u8],
+        ) -> Poll<Result<usize, Error>> {
+            let size: usize = cmp::min(self.read_data.len(), buf.len());
+            buf[..size].copy_from_slice(&self.read_data[..size]);
+            Poll::Ready(Ok(size))
+        }
+    }
+
+    impl Write for MockTcpStream {
+        fn poll_write(
+            mut self: Pin<&mut Self>,
+            _: &mut Context,
+            buf: &[u8],
+        ) -> Poll<Result<usize, Error>> {
+            self.write_data = Vec::from(buf);
+
+            Poll::Ready(Ok(buf.len()))
+        }
+
+        fn poll_flush(self: Pin<&mut Self>, _: &mut Context) -> Poll<Result<(), Error>> {
+            Poll::Ready(Ok(()))
+        }
+
+        fn poll_close(self: Pin<&mut Self>, _: &mut Context) -> Poll<Result<(), Error>> {
+            Poll::Ready(Ok(()))
+        }
+    }
+
+    impl Unpin for MockTcpStream {}
+
+    #[async_std::test]
+    async fn test_handle_connection() {
+        let input_bytes = b"GET / HTTP/1.1\r\n";
+        let mut contents = vec![0u8; 1024];
+        contents[..input_bytes.len()].clone_from_slice(input_bytes);
+        let mut stream = MockTcpStream {
+            read_data: contents,
+            write_data: Vec::new(),
+        };
+
+        handle_connection(&mut stream).await;
+
+        let expected_contents = fs::read_to_string("index.html").unwrap();
+        let expected_response = format!("HTTP/1.1 200 OK\r\n\r\n{}", expected_contents);
+        assert!(stream.write_data.starts_with(expected_response.as_bytes()));
+    }
+}