async-http: Serve requests concurrently
authorMartin Pitt <martin@piware.de>
Fri, 16 Sep 2022 06:47:48 +0000 (08:47 +0200)
committerMartin Pitt <martin@piware.de>
Fri, 16 Sep 2022 06:58:56 +0000 (08:58 +0200)
Move to asyncstd TCPListener and futures Stream, so that the incoming
loop does not serialize/block requests any more.

This is still single-threaded. That can be demonstrated with replacing
the async sleep with a sync one (commented out). Then /sleep will block
other requests again.

async-http/Cargo.toml
async-http/src/main.rs

index 92c94f55844386e63136b65f558b21b19e75612d..30b493032f952a848eb31be06095d5e46f3b0239 100644 (file)
@@ -6,6 +6,7 @@ edition = "2021"
 # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
 
 [dependencies]
+futures = "0.3"
 log = "0.4"
 env_logger = "0.9"
 
index c36be0ad86d093e4619b1c3e41dfe474d4d5a073..cf0b46de8e2995b0da941ce29f0609bf540571cb 100644 (file)
@@ -1,29 +1,28 @@
 use std::fs;
-use std::io::prelude::*;
-use std::net::{ TcpListener, TcpStream };
 use std::time::Duration;
 
+use async_std::prelude::*;
+use async_std::net::{ TcpListener, TcpStream };
 use async_std::task;
+use futures::stream::StreamExt;
 
 #[async_std::main]
 async fn main() {
     env_logger::Builder::from_env(env_logger::Env::default().default_filter_or("info")).init();
 
     // Listen for incoming TCP connections on localhost port 7878
-    let listener = TcpListener::bind("127.0.0.1:7878").unwrap();
+    let listener = TcpListener::bind("127.0.0.1:7878").await.unwrap();
 
-    // Block forever, handling each request that arrives at this IP address
-    for stream in listener.incoming() {
-        let stream = stream.unwrap();
-        // not concurrent
-        handle_connection(stream).await;
-    }
+    listener.incoming().for_each_concurrent(/* limit */ None, |tcpstream| async move {
+        let tcpstream = tcpstream.unwrap();
+        handle_connection(tcpstream).await;
+    }).await;
 }
 
 async fn handle_connection(mut stream: TcpStream) {
     // Read the first 1024 bytes of data from the stream
     let mut buffer = [0; 1024];
-    assert!(stream.read(&mut buffer).unwrap() > 0);
+    assert!(stream.read(&mut buffer).await.unwrap() > 0);
 
     // Respond with greetings or a 404,
     // depending on the data in the request
@@ -31,6 +30,8 @@ async fn handle_connection(mut stream: TcpStream) {
         ("HTTP/1.1 200 OK", "index.html")
     } else if buffer.starts_with(b"GET /sleep HTTP/1.1\r\n") {
         task::sleep(Duration::from_secs(5)).await;
+        // sync version, to demonstrate concurrent async vs. parallel threads
+        // std::thread::sleep(Duration::from_secs(5));
         ("HTTP/1.1 201 Sleep", "index.html")
     } else {
         ("HTTP/1.1 404 NOT FOUND", "404.html")
@@ -41,6 +42,6 @@ async fn handle_connection(mut stream: TcpStream) {
     // Write response back to the stream,
     // and flush the stream to ensure the response is sent back to the client
     let response = format!("{status_line}\r\n\r\n{contents}");
-    stream.write_all(response.as_bytes()).unwrap();
-    stream.flush().unwrap();
+    stream.write_all(response.as_bytes()).await.unwrap();
+    stream.flush().await.unwrap();
 }