Info
This is the first post where I’m doing “quick bytes” of simple examples with Rust. This is both a way for me to keep track of some basic usage of various Rust-isms, but also a way for me to share that in case it’s helpful. I could just leave these as notes in my Obsidian vault, but why not share!
UDP
The User Datagram Protocol (UDP) is the lesser known sibling of the Transmission Control Protocol (TCP). The simple difference is TCP is guaranteed and UDP is not. What I mean is, UDP is like “fire and forget” and TCP has a whole retry mechanism. There are of course other differences, but that’s what I see as key to understand. UDP is also used for “multicast” where a packet is broadcast to multiple recipients.
UDP in Rust
Rust implements UDP in the standard library with std::net::UdpSocket
. You
bind the socket to an IP address and port, and then you can send and receive
data on that port at that address. If you want to send data to another UDP
socket, you bind to that address and port, then connect one to the other.
Obviously the typical usage would be for sending messages between processes. In
this simple “quick byte” example though, I will just send
a message between threads in the same process. Ignoring the threading aspects
and focusing on the UDP parts, notice that I’m using 0
as the port in the
IP address: 127.0.0.1:0
. This tells the OS to pick a random port for me (and
of course 127.0.0.1
is the localhost address). Because the port is randomly
assigned, I use the local_addr()
method to find the port of the “server”
so I can connect the udp_client
to the udp_server
. Everything else should be
pretty self-explanatory … use recv_from(&mut buf)
to read data and use
send_to(&buf, addr)
to send data. The receive method doesn’t require an address
because it reads from the port it’s bound to. However, sending data requires the
address where to send, since that can obviously be anywhere.
Important
For the simple example I just return errors from the
main()
method and use the?
shortcut to return errors. Naturally in a “real” program you would use error handling best practices.
use std::{error::Error, net::UdpSocket, thread};
fn main() -> Result<(), Box<dyn Error>> {
let udp_client = UdpSocket::bind("127.0.0.1:0")?;
let udp_server = UdpSocket::bind("127.0.0.1:0")?;
let addr = udp_server.local_addr()?;
udp_client.connect(addr)?;
println!("client: {:?}", udp_client);
println!("server: {:?}", udp_server);
thread::spawn(move || {
let mut buf = [0u8; 1024];
let (sz, addr) = udp_server
.recv_from(&mut buf)
.expect("error reading from client");
println!("server: read {sz} bytes");
udp_server.send_to(&buf[..sz], addr)
});
let sz = udp_client.send_to("hello, world".as_bytes(), addr)?;
println!("client: wrote {sz} bytes");
let mut buf = [0u8; 1024];
let (sz, _addr) = udp_client
.recv_from(&mut buf)
.expect("error reading from client");
println!("client: read {sz} bytes");
Ok(())
}
And there you have it. Since this is the first “quick byte” please let me know if you like this format.