This crate provides an extension trait for std::net::UdpSocket that supports source address
selection for outgoing UDP datagrams. This is useful for implementing a UDP server that binds
multiple network interfaces.
The implementation relies on socket options IP_PKTINFO (for IPv4) and IPV6_RECVPKTINFO
(for IPv6).
use std::net::{UdpSocket,SocketAddr};
use udp_sas::UdpSas;
fn main() {
demo().unwrap();
}
fn demo() -> std::io::Result<()>
{
let mut buf = [0u8;128];
let srv = UdpSocket::bind_sas("0.0.0.0:30012".parse::<SocketAddr>().unwrap())?;
let srv_addr = "127.0.0.23:30012".parse().unwrap();
let cli = UdpSocket::bind_sas("0.0.0.0:0".parse::<SocketAddr>().unwrap())?;
let cli_addr = SocketAddr::new(
"127.0.0.45".parse().unwrap(),
cli.local_addr().unwrap().port());
assert_ne!(cli_addr.port(), 0);
let msg1 = "What do you get if you multiply six by nine?";
let nb = cli.send_sas(msg1.as_bytes(), &srv_addr, &cli_addr.ip())?;
assert_eq!(nb, msg1.as_bytes().len());
let (nb, peer, local) = srv.recv_sas(&mut buf)?;
assert_eq!(peer, cli_addr);
assert_eq!(local, srv_addr.ip());
assert_eq!(nb, msg1.as_bytes().len());
assert_eq!(&buf[0..nb], msg1.as_bytes());
let msg2 = "Forty-two";
let nb = srv.send_sas(msg2.as_bytes(), &peer, &local)?;
assert_eq!(nb, msg2.as_bytes().len());
let (nb, peer, local) = cli.recv_sas(&mut buf)?;
assert_eq!(peer, srv_addr);
assert_eq!(local, cli_addr.ip());
assert_eq!(nb, msg2.as_bytes().len());
assert_eq!(&buf[0..nb], msg2.as_bytes());
Ok(())
}