Receiving messages
Let's implement the receiving part of the protocol. We need to:
- split incoming
TcpStream
on\n
and decode bytes as utf-8 - interpret the first line as a login
- parse the rest of the lines as a
login: message
We highly recommend to go past this quick, this is a lot of protocol minutia.
extern crate tokio;
use std::{
collections::hash_map::{Entry, HashMap},
future::Future,
};
use tokio::{
io::{AsyncBufReadExt, AsyncWriteExt, BufReader},
net::{tcp::OwnedWriteHalf, TcpListener, TcpStream, ToSocketAddrs},
sync::{mpsc, oneshot},
task,
};
type Result<T> = std::result::Result<T, Box<dyn std::error::Error + Send + Sync>>;
async fn accept_loop(addr: impl ToSocketAddrs) -> Result<()> {
let listener = TcpListener::bind(addr).await?;
loop {
let (stream, _socket_addr) = listener.accept().await?;
println!("Accepting from: {}", stream.peer_addr()?);
let _handle = task::spawn(connection_loop(stream));
}
Ok(())
}
async fn connection_loop(stream: TcpStream) -> Result<()> {
let reader = BufReader::new(stream);
let mut lines = reader.lines(); // 2
// 3
let name = match lines.next_line().await? {
None => Err("peer disconnected immediately")?,
Some(line) => line,
};
println!("name = {}", name);
// 4
loop {
if let Some(line) = lines.next_line().await? {
// 5
let (dest, msg) = match line.find(':') {
None => continue,
Some(idx) => (&line[..idx], line[idx + 1..].trim()),
};
let dest = dest
.split(',')
.map(|name| name.trim().to_string())
.collect::<Vec<_>>();
let msg = msg.to_string();
// TODO: this is temporary
println!("Received message: {}", msg);
} else {
break
}
}
Ok(())
}
-
We use
task::spawn
function to spawn an independent task for working with each client. That is, after accepting the client theaccept_loop
immediately starts waiting for the next one. This is the core benefit of event-driven architecture: we serve many clients concurrently, without spending many hardware threads. -
Luckily, the "split byte stream into lines" functionality is already implemented.
.lines()
call returns a stream ofString
's. -
We get the first line -- login
-
And, once again, we implement a manual async loop.
-
Finally, we parse each line into a list of destination logins and the message itself.
Managing Errors
One serious problem in the above solution is that, while we correctly propagate errors in the connection_loop
, we just drop the error on the floor afterwards!
That is, task::spawn
does not return an error immediately (it can't, it needs to run the future to completion first), only after it is joined.
We can "fix" it by waiting for the task to be joined, like this:
extern crate tokio;
use tokio::{
net::TcpStream,
task,
};
type Result<T> = std::result::Result<T, Box<dyn std::error::Error + Send + Sync>>;
async fn connection_loop(stream: TcpStream) -> Result<()> {
Ok(())
}
async fn accept_loop(stream: TcpStream) -> Result<()> {
let handle = task::spawn(connection_loop(stream));
handle.await?
}
The .await
waits until the client finishes, and ?
propagates the result.
There are two problems with this solution however! First, because we immediately await the client, we can only handle one client at a time, and that completely defeats the purpose of async! Second, if a client encounters an IO error, the whole server immediately exits. That is, a flaky internet connection of one peer brings down the whole chat room!
A correct way to handle client errors in this case is log them, and continue serving other clients. So let's use a helper function for this:
extern crate tokio;
use std::future::Future;
use tokio::task;
type Result<T> = std::result::Result<T, Box<dyn std::error::Error + Send + Sync>>;
fn spawn_and_log_error<F>(fut: F) -> task::JoinHandle<()>
where
F: Future<Output = Result<()>> + Send + 'static,
{
task::spawn(async move {
if let Err(e) = fut.await {
eprintln!("{}", e)
}
})
}