#![cfg_attr(
not(any(feature = "default-resolver", feature = "ring-accelerated",)),
allow(dead_code, unused_extern_crates, unused_imports)
)]
use hex::FromHex;
use snow::{Builder, Keypair, params::NoiseParams};
use std::{
io::{self, Read, Write},
net::{TcpListener, TcpStream},
sync::LazyLock,
};
static SECRET: &[u8; 32] = b"i don't care for fidget spinners";
static PARAMS: LazyLock<NoiseParams> =
LazyLock::new(|| "Noise_Xpsk1_25519_ChaChaPoly_BLAKE2s".parse().unwrap());
static RESPONDER: LazyLock<Keypair> = LazyLock::new(|| Keypair {
private: Vec::from_hex("52fbe3721d1adbe312d270ca2db5ce5bd39ddc206075f3a8f06d422619c8eb5d")
.expect("valid hex"),
public: Vec::from_hex("435ce8a8415ccd44de5e207581ac7207b416683028bcaecc9eb38d944e6f900c")
.expect("valid hex"),
});
#[cfg(any(feature = "default-resolver", feature = "ring-accelerated"))]
fn main() {
let server_mode =
std::env::args().next_back().is_none_or(|arg| arg == "-s" || arg == "--server");
if server_mode {
run_server(&RESPONDER.private, SECRET);
} else {
run_client(&RESPONDER.public, SECRET);
}
println!("all done.");
}
#[cfg(any(feature = "default-resolver", feature = "ring-accelerated"))]
fn run_server(private_key: &[u8], psk: &[u8; 32]) {
let mut buf = vec![0_u8; 65535];
let builder = Builder::new(PARAMS.clone());
let mut noise = builder
.local_private_key(private_key)
.unwrap()
.psk(1, psk)
.unwrap()
.build_responder()
.unwrap();
println!("listening on 127.0.0.1:9999");
let (mut stream, _) = TcpListener::bind("127.0.0.1:9999").unwrap().accept().unwrap();
noise.read_message(&recv(&mut stream).unwrap(), &mut buf).unwrap();
let client = hex::encode(noise.get_remote_static().unwrap());
let mut transport = noise.into_transport_mode().unwrap();
while let Ok(msg) = recv(&mut stream) {
let len = transport.read_message(&msg, &mut buf).unwrap();
println!("{client} said: {}", String::from_utf8_lossy(&buf[..len]));
}
println!("connection closed.");
}
#[cfg(any(feature = "default-resolver", feature = "ring-accelerated"))]
fn run_client(responder_public_key: &[u8], psk: &[u8; 32]) {
let mut buf = vec![0_u8; 65535];
let builder = Builder::new(PARAMS.clone());
let private_key = &builder.generate_keypair().unwrap().private;
let mut noise = builder
.local_private_key(private_key)
.unwrap()
.remote_public_key(responder_public_key)
.unwrap()
.psk(1, psk)
.unwrap()
.build_initiator()
.unwrap();
let mut stream = TcpStream::connect("127.0.0.1:9999").unwrap();
println!("connected...");
let len = noise.write_message(&[], &mut buf).unwrap();
send(&mut stream, &buf[..len]);
let mut transport = noise.into_transport_mode().unwrap();
println!("session established...");
for _ in 0..10 {
let len = transport.write_message(b"HACK THE PLANET", &mut buf).unwrap();
send(&mut stream, &buf[..len]);
}
println!("notified server of intent to hack planet.");
}
fn recv(stream: &mut TcpStream) -> io::Result<Vec<u8>> {
let mut msg_len_buf = [0_u8; 2];
stream.read_exact(&mut msg_len_buf)?;
let msg_len = usize::from(u16::from_be_bytes(msg_len_buf));
let mut msg = vec![0_u8; msg_len];
stream.read_exact(&mut msg[..])?;
Ok(msg)
}
fn send(stream: &mut TcpStream, buf: &[u8]) {
let len = u16::try_from(buf.len()).expect("message too large");
stream.write_all(&len.to_be_bytes()).unwrap();
stream.write_all(buf).unwrap();
}
#[cfg(not(any(feature = "default-resolver", feature = "ring-accelerated")))]
fn main() {
panic!("Example must be compiled with some cryptographic provider.");
}