summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authorDavid Lönnhager <david.l@mullvad.net>2025-04-05 00:40:05 +0200
committerDavid Lönnhager <david.l@mullvad.net>2025-04-07 15:52:23 +0200
commitb448f45b0bc13c367bbd4bb272431682f8a65cc4 (patch)
treec037ea9f2d612cb5ef237ad185ee6a3bc86c1c1e
parent9938127079922a261ecce358f67299dfe92214e8 (diff)
downloadmullvadvpn-b448f45b0bc13c367bbd4bb272431682f8a65cc4.tar.xz
mullvadvpn-b448f45b0bc13c367bbd4bb272431682f8a65cc4.zip
Add end-to-end test for MASQUE proxy
-rw-r--r--Cargo.lock1
-rw-r--r--mullvad-masque-proxy/Cargo.toml3
-rw-r--r--mullvad-masque-proxy/tests/proxy.rs111
-rw-r--r--mullvad-masque-proxy/tests/test.crt22
-rw-r--r--mullvad-masque-proxy/tests/test.key28
5 files changed, 164 insertions, 1 deletions
diff --git a/Cargo.lock b/Cargo.lock
index b29f46eebc..8a377950dd 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -2871,6 +2871,7 @@ dependencies = [
name = "mullvad-masque-proxy"
version = "0.1.0"
dependencies = [
+ "anyhow",
"bytes",
"clap",
"h3",
diff --git a/mullvad-masque-proxy/Cargo.toml b/mullvad-masque-proxy/Cargo.toml
index e6307eb3f4..27b1a9ca1f 100644
--- a/mullvad-masque-proxy/Cargo.toml
+++ b/mullvad-masque-proxy/Cargo.toml
@@ -21,7 +21,8 @@ rustls-pemfile = "2.1.3"
bytes = "1"
[dev-dependencies]
-tokio = { workspace = true, features = [ "macros", "io-util", "rt-multi-thread" ] }
+anyhow = { workspace = true }
+tokio = { workspace = true, features = ["fs", "macros", "io-util", "rt-multi-thread"] }
clap = { workspace = true }
rand = "0.8.5"
diff --git a/mullvad-masque-proxy/tests/proxy.rs b/mullvad-masque-proxy/tests/proxy.rs
new file mode 100644
index 0000000000..6825f7d75d
--- /dev/null
+++ b/mullvad-masque-proxy/tests/proxy.rs
@@ -0,0 +1,111 @@
+use std::net::SocketAddr;
+use std::sync::Arc;
+
+use anyhow::Context;
+use bytes::BytesMut;
+use tokio::fs;
+
+use mullvad_masque_proxy::client;
+use mullvad_masque_proxy::server;
+use tokio::net::UdpSocket;
+
+/// Set up a MASQUE proxy and test that it can be used to communicate with some UDP destination
+#[tokio::test]
+async fn test_server_and_client_forwarding() -> anyhow::Result<()> {
+ const MAXIMUM_PACKET_SIZE: u16 = 1700;
+ const HOST: &str = "test.test";
+
+ let any_localhost_addr: SocketAddr = "127.0.0.1:0".parse().unwrap();
+
+ // Set up destination UDP server
+ let target_udp_server = UdpSocket::bind(any_localhost_addr).await?;
+ let target_udp_addr = target_udp_server
+ .local_addr()
+ .context("Retrieve dest UDP server addr")?;
+
+ // Set up MASQUE server
+ let server_tls_config = load_server_test_cert().await?;
+ let server = server::Server::bind(
+ any_localhost_addr,
+ Default::default(),
+ Arc::new(server_tls_config),
+ MAXIMUM_PACKET_SIZE,
+ )
+ .context("Failed to start MASQUE server")?;
+
+ let masque_server_addr = server.local_addr()?;
+
+ tokio::spawn(server.run());
+
+ // Set up MASQUE client
+ let local_socket = UdpSocket::bind(any_localhost_addr)
+ .await
+ .context("Failed to bind address")?;
+ let masque_client_addr = local_socket.local_addr().unwrap();
+
+ let client = client::Client::connect_with_tls_config(
+ local_socket,
+ masque_server_addr,
+ // Local QUIC address
+ any_localhost_addr,
+ target_udp_addr,
+ HOST,
+ client::default_tls_config(),
+ MAXIMUM_PACKET_SIZE,
+ )
+ .await
+ .context("Failed to start MASQUE client")?;
+
+ tokio::spawn(client.run());
+
+ // Connect to local UDP socket
+ let proxy_client = UdpSocket::bind(any_localhost_addr).await?;
+ proxy_client
+ .connect(masque_client_addr)
+ .await
+ .context("Failed to connect to local UDP server")?;
+
+ // Proxy client -> destination
+ let mut rx_buf = BytesMut::with_capacity(128);
+ proxy_client.send(b"abc").await?;
+ let (_, proxy_addr) = target_udp_server
+ .recv_buf_from(&mut rx_buf)
+ .await
+ .context("Expected to receive message")?;
+ assert_eq!(&*rx_buf, b"abc", "Expected to receive message from client");
+
+ // Destination -> proxy client
+ let mut rx_buf = BytesMut::with_capacity(128);
+ target_udp_server.send_to(b"def", proxy_addr).await?;
+ proxy_client
+ .recv_buf(&mut rx_buf)
+ .await
+ .context("Expected to receive message")?;
+ assert_eq!(&*rx_buf, b"def", "Expected to receive message from server");
+
+ Ok(())
+}
+
+async fn load_server_test_cert() -> anyhow::Result<rustls::ServerConfig> {
+ let key = fs::read("tests/test.key").await.context("Read test key")?;
+ let key = rustls_pemfile::private_key(&mut &*key)?.context("Invalid test key")?;
+
+ let cert_chain = fs::read("tests/test.crt")
+ .await
+ .context("Read test certificate")?;
+ let cert_chain = rustls_pemfile::certs(&mut &*cert_chain)
+ .collect::<Result<_, _>>()
+ .context("Invalid test certificate")?;
+
+ let mut tls_config = rustls::ServerConfig::builder_with_provider(Arc::new(
+ rustls::crypto::ring::default_provider(),
+ ))
+ .with_protocol_versions(&[&rustls::version::TLS13])?
+ .with_no_client_auth()
+ .with_single_cert(cert_chain, key)?;
+
+ tls_config.max_early_data_size = u32::MAX;
+ tls_config.alpn_protocols = vec![b"h3".into()];
+
+ Ok(tls_config)
+}
diff --git a/mullvad-masque-proxy/tests/test.crt b/mullvad-masque-proxy/tests/test.crt
new file mode 100644
index 0000000000..2cc992fd24
--- /dev/null
+++ b/mullvad-masque-proxy/tests/test.crt
@@ -0,0 +1,22 @@
+-----BEGIN CERTIFICATE-----
+MIIDkzCCAnugAwIBAgIUC2y+OPgQ2HbWRXTZwjvZOFlGRcIwDQYJKoZIhvcNAQEL
+BQAwWTELMAkGA1UEBhMCQVUxEzARBgNVBAgMClNvbWUtU3RhdGUxITAfBgNVBAoM
+GEludGVybmV0IFdpZGdpdHMgUHR5IEx0ZDESMBAGA1UEAwwJdGVzdC50ZXN0MB4X
+DTI1MDQwMzExNDUxNVoXDTI1MDUwMzExNDUxNVowWTELMAkGA1UEBhMCQVUxEzAR
+BgNVBAgMClNvbWUtU3RhdGUxITAfBgNVBAoMGEludGVybmV0IFdpZGdpdHMgUHR5
+IEx0ZDESMBAGA1UEAwwJdGVzdC50ZXN0MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8A
+MIIBCgKCAQEAufg598upax+qDkcQCvy4Gv9RSD04JMMnYIa6wgGSZcKuur4r9FRr
+b+pnmiwwoZPlmtE5GUnvVVXB+UGr889IatQ00HxfTr8A/McHh/bSg43uEVnLBWDy
+S9ULW6gJX6xwCvZINRCyw6B6eEBOs2T1MCsaA2/x4ba0lYT8lT9ApRCZE1HFh/Mb
+c1/N/VUaTTxtsE/mhg8sYc2ig1EcZMDy/b7pPUz83oUa4zHruqNQUb2bFQ0TrcKY
+C456aJXCb6t5rWiUJmByzjFOwVGokE550km1q7dpv3yjK5Z023O2jnBVfpVCeEAA
+mnsPQAgqFpuXXTHPqBvuyzHzjsWkZN5wXwIDAQABo1MwUTAdBgNVHQ4EFgQUEEaP
+f5wUEnKp5WZTiMdoydTPq5kwHwYDVR0jBBgwFoAUEEaPf5wUEnKp5WZTiMdoydTP
+q5kwDwYDVR0TAQH/BAUwAwEB/zANBgkqhkiG9w0BAQsFAAOCAQEAPrJfHriKFeLn
+hV9lrH+EN/Q62q1ZIxQagDadcBjhfMALfsD2nr5ENL3uVvFz+XrB9pVrMlfBTnx/
++rbzlzfTpIgKHXsICnXQbY8cvA54JyXysHf9jP3yvZP4vlCjTYX8JuWlykm4StMZ
+suPWyCYprCfJd0n8T2heFyWl6Q9MeyrQolpkrhhR6JuifH/ySc4dl3yWEMrX7lEe
+435llV5WolhiyQ060tKFfgrfPu245DO7Mci9QmafKGK4WSYil95Cwy1uYevVtiNa
+oZQ9mjWJJvVab17qTe9lWukMHFDr7qCxtwTWxynDhWvLrvlPP/2DKo0nEGZU/cVV
+ozh88D0xZw==
+-----END CERTIFICATE-----
diff --git a/mullvad-masque-proxy/tests/test.key b/mullvad-masque-proxy/tests/test.key
new file mode 100644
index 0000000000..aff1bc2dae
--- /dev/null
+++ b/mullvad-masque-proxy/tests/test.key
@@ -0,0 +1,28 @@
+-----BEGIN PRIVATE KEY-----
+MIIEvAIBADANBgkqhkiG9w0BAQEFAASCBKYwggSiAgEAAoIBAQC5+Dn3y6lrH6oO
+RxAK/Lga/1FIPTgkwydghrrCAZJlwq66viv0VGtv6meaLDChk+Wa0TkZSe9VVcH5
+Qavzz0hq1DTQfF9OvwD8xweH9tKDje4RWcsFYPJL1QtbqAlfrHAK9kg1ELLDoHp4
+QE6zZPUwKxoDb/HhtrSVhPyVP0ClEJkTUcWH8xtzX839VRpNPG2wT+aGDyxhzaKD
+URxkwPL9vuk9TPzehRrjMeu6o1BRvZsVDROtwpgLjnpolcJvq3mtaJQmYHLOMU7B
+UaiQTnnSSbWrt2m/fKMrlnTbc7aOcFV+lUJ4QACaew9ACCoWm5ddMc+oG+7LMfOO
+xaRk3nBfAgMBAAECggEAI/G5ao6fuUfOe6H6lNUR1I4CrN7ASkK6Cqsfz720CR0e
+3pNBNaFXfrMkwSTHZYOLfmfwDFZA/xJrQn0R+jbXPWa0qpNPbI34Z+MkLoBjYe/9
+0razSd/aFRQhdN6+qRJQOZ4uiKsoki0jXri3PW9HAL9j8MQjUUgaEUg59bLbEMwX
+IR98n+aKLSFBJeeUKE5tvLPt9yvegNI1/6nELT7My3SnIOMlVN9njQq8wqqIch2X
+ky7NwlfsCZ1hF1sD8xCUBloUo/LKfCxWm+2Kt0YQ0/lYfwp8WzFXt4XAL50AKW+A
+QrzVpNMwIp/k/FeKcMF/6rbz3iqQTKW2RrK9rX08WQKBgQDbluIh36xLbSFDjLNx
+j9dCiiCI3cQZVBc7FTfStc9/k4O1NBq5YkUZlCJzjbFsxakdRcb4O2Kwf/v8sBq3
+C9PmoJVV/gqzJQ8TjLJF0MPNPzynm7B0YAD4RXUODRymkqFyHBa3uXT7WG6CmdEa
+Ggh8y5xlKZ2c19J6WpQi6EwM8wKBgQDYzkD/7ad5ttSdVUXNr57Oe9JONwUDUPT6
+UaCthUAbeeKM6JeSZ658DikKALMDZ73grncjSQgLPtOSvo/LQxda+AG515ZZ7J5T
+Nauj44UYFz22Ck7guZCq5tQQVx4YUlBRtM7El9oOvYLjBaLTl9lmmCeQGj4+ugl2
+FU/qDGh55QKBgAlARQyaSL7wvQsEfXbWUYJLIW3CsgVDJqtljHGDGVfNlinnJQ0U
+V8bpF754hLYJacOC8gv5LII1Eh+mJ6n4hJfdwgzaZAcCE62GKuiIEAewl1SUWY29
+kazj+Dd8U+2slcKh7k8VMBl6s0UrR8TqvdrMFS2p4CsAaKyg7ka+NJ4DAoGAMZy5
+KRekLGkXLE24JIJcr9mL3ZQflIuxE5scTrjgW6k/m4kaLkmFlyPSZlSUomHaBJFH
++A4dRh2BYuIym4vly05XbsSTxk4sSNROS7mj2khvOboQJMKyBTm/K2IUI/KqKJhc
+fIZXQupBClxez1a/TAfjfclTlx0RTzE/UUq3mbUCgYBGqeq9cgs9qxt0WS0+Un3o
+/cgGXLknk4IsVclMgPPiDhujTs3I/dJu94zwY3dch906k5KKctYX7cS3YUhAMzBX
+e2DYRq/Z1JC9OXCr7HLTF60IjFx7KlGRL/Kmx1Dd98IZmhS2RSE+dJPovlsYqZ1n
+dGHHcZoEs4iiPe6m+umVyw==
+-----END PRIVATE KEY-----