Implementing a client
Since the protocol is line-based, implementing a client for the chat is straightforward:
- Lines read from stdin should be sent over the socket.
- Lines read from the socket should be echoed to stdout.
Although async does not significantly affect client performance (as unlike the server, the client interacts solely with one user and only needs limited concurrency), async is still useful for managing concurrency!
The client has to read from stdin and the socket simultaneously.
Programming this with threads is cumbersome, especially when implementing a clean shutdown.
With async, the select!
macro is all that is needed.
extern crate tokio;
use tokio::{
io::{stdin, AsyncBufReadExt, AsyncWriteExt, BufReader},
net::{TcpStream, ToSocketAddrs},
};
type Result<T> = std::result::Result<T, Box<dyn std::error::Error + Send + Sync>>;
// main
async fn run() -> Result<()> {
try_main("127.0.0.1:8080").await
}
async fn try_main(addr: impl ToSocketAddrs) -> Result<()> {
let stream = TcpStream::connect(addr).await?;
let (reader, mut writer) = stream.into_split();
let mut lines_from_server = BufReader::new(reader).lines(); // 2
let mut lines_from_stdin = BufReader::new(stdin()).lines(); // 3
loop {
tokio::select! { // 4
line = lines_from_server.next_line() => match line {
Ok(Some(line)) => {
println!("{}", line);
},
Ok(None) => break,
Err(e) => eprintln!("Error {:?}:", e),
},
line = lines_from_stdin.next_line() => match line {
Ok(Some(line)) => {
writer.write_all(line.as_bytes()).await?;
writer.write_all(b"\n").await?;
},
Ok(None) => break,
Err(e) => eprintln!("Error {:?}:", e),
}
}
}
Ok(())
}
- Here we split
TcpStream
into read and write halves. - We create a stream of lines for the socket.
- We create a stream of lines for stdin.
- In the main select loop, we print the lines we receive from the server and send the lines we read from the console.