forked from solana-labs/solana
-
Notifications
You must be signed in to change notification settings - Fork 265
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
move server/client configuration for async quic tests to separate unit (
#2219) Move server/client configuration functions from unit tests to separate unit. This allows to use them in the integration tests.
- Loading branch information
1 parent
1519a2e
commit 97b452f
Showing
5 changed files
with
169 additions
and
145 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -5,3 +5,4 @@ pub mod rate_limiter; | |
pub mod recvmmsg; | ||
pub mod sendmmsg; | ||
mod stream_throttle; | ||
pub mod testing_utilities; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,161 @@ | ||
//! Contains utility functions to create server and client for test purposes. | ||
use { | ||
super::quic::{ | ||
spawn_server_multi, SpawnNonBlockingServerResult, ALPN_TPU_PROTOCOL_ID, | ||
DEFAULT_MAX_CONNECTIONS_PER_IPADDR_PER_MINUTE, DEFAULT_MAX_STREAMS_PER_MS, | ||
}, | ||
crate::{ | ||
quic::{StreamerStats, MAX_STAKED_CONNECTIONS, MAX_UNSTAKED_CONNECTIONS}, | ||
streamer::StakedNodes, | ||
tls_certificates::new_dummy_x509_certificate, | ||
}, | ||
crossbeam_channel::unbounded, | ||
quinn::{ClientConfig, Connection, EndpointConfig, IdleTimeout, TokioRuntime, TransportConfig}, | ||
solana_perf::packet::PacketBatch, | ||
solana_sdk::{ | ||
net::DEFAULT_TPU_COALESCE, | ||
quic::{QUIC_KEEP_ALIVE, QUIC_MAX_TIMEOUT}, | ||
signer::keypair::Keypair, | ||
}, | ||
std::{ | ||
net::{SocketAddr, UdpSocket}, | ||
sync::{atomic::AtomicBool, Arc, RwLock}, | ||
time::Duration, | ||
}, | ||
tokio::task::JoinHandle, | ||
}; | ||
|
||
struct SkipServerVerification; | ||
|
||
impl SkipServerVerification { | ||
fn new() -> Arc<Self> { | ||
Arc::new(Self) | ||
} | ||
} | ||
|
||
impl rustls::client::ServerCertVerifier for SkipServerVerification { | ||
fn verify_server_cert( | ||
&self, | ||
_end_entity: &rustls::Certificate, | ||
_intermediates: &[rustls::Certificate], | ||
_server_name: &rustls::ServerName, | ||
_scts: &mut dyn Iterator<Item = &[u8]>, | ||
_ocsp_response: &[u8], | ||
_now: std::time::SystemTime, | ||
) -> Result<rustls::client::ServerCertVerified, rustls::Error> { | ||
Ok(rustls::client::ServerCertVerified::assertion()) | ||
} | ||
} | ||
|
||
pub fn get_client_config(keypair: &Keypair) -> ClientConfig { | ||
let (cert, key) = new_dummy_x509_certificate(keypair); | ||
|
||
let mut crypto = rustls::ClientConfig::builder() | ||
.with_safe_defaults() | ||
.with_custom_certificate_verifier(SkipServerVerification::new()) | ||
.with_client_auth_cert(vec![cert], key) | ||
.expect("Failed to use client certificate"); | ||
|
||
crypto.enable_early_data = true; | ||
crypto.alpn_protocols = vec![ALPN_TPU_PROTOCOL_ID.to_vec()]; | ||
|
||
let mut config = ClientConfig::new(Arc::new(crypto)); | ||
|
||
let mut transport_config = TransportConfig::default(); | ||
let timeout = IdleTimeout::try_from(QUIC_MAX_TIMEOUT).unwrap(); | ||
transport_config.max_idle_timeout(Some(timeout)); | ||
transport_config.keep_alive_interval(Some(QUIC_KEEP_ALIVE)); | ||
config.transport_config(Arc::new(transport_config)); | ||
|
||
config | ||
} | ||
|
||
pub fn setup_quic_server( | ||
option_staked_nodes: Option<StakedNodes>, | ||
max_connections_per_peer: usize, | ||
) -> ( | ||
JoinHandle<()>, | ||
Arc<AtomicBool>, | ||
crossbeam_channel::Receiver<PacketBatch>, | ||
SocketAddr, | ||
Arc<StreamerStats>, | ||
) { | ||
let sockets = { | ||
#[cfg(not(target_os = "windows"))] | ||
{ | ||
use std::{ | ||
os::fd::{FromRawFd, IntoRawFd}, | ||
str::FromStr as _, | ||
}; | ||
(0..10) | ||
.map(|_| { | ||
let sock = socket2::Socket::new( | ||
socket2::Domain::IPV4, | ||
socket2::Type::DGRAM, | ||
Some(socket2::Protocol::UDP), | ||
) | ||
.unwrap(); | ||
sock.set_reuse_port(true).unwrap(); | ||
sock.bind(&SocketAddr::from_str("127.0.0.1:0").unwrap().into()) | ||
.unwrap(); | ||
unsafe { UdpSocket::from_raw_fd(sock.into_raw_fd()) } | ||
}) | ||
.collect::<Vec<_>>() | ||
} | ||
#[cfg(target_os = "windows")] | ||
{ | ||
vec![UdpSocket::bind("127.0.0.1:0").unwrap()] | ||
} | ||
}; | ||
|
||
let exit = Arc::new(AtomicBool::new(false)); | ||
let (sender, receiver) = unbounded(); | ||
let keypair = Keypair::new(); | ||
let server_address = sockets[0].local_addr().unwrap(); | ||
let staked_nodes = Arc::new(RwLock::new(option_staked_nodes.unwrap_or_default())); | ||
let SpawnNonBlockingServerResult { | ||
endpoints: _, | ||
stats, | ||
thread: t, | ||
max_concurrent_connections: _, | ||
} = spawn_server_multi( | ||
"quic_streamer_test", | ||
sockets, | ||
&keypair, | ||
sender, | ||
exit.clone(), | ||
max_connections_per_peer, | ||
staked_nodes, | ||
MAX_STAKED_CONNECTIONS, | ||
MAX_UNSTAKED_CONNECTIONS, | ||
DEFAULT_MAX_STREAMS_PER_MS, | ||
DEFAULT_MAX_CONNECTIONS_PER_IPADDR_PER_MINUTE, | ||
Duration::from_secs(2), | ||
DEFAULT_TPU_COALESCE, | ||
) | ||
.unwrap(); | ||
(t, exit, receiver, server_address, stats) | ||
} | ||
|
||
pub async fn make_client_endpoint( | ||
addr: &SocketAddr, | ||
client_keypair: Option<&Keypair>, | ||
) -> Connection { | ||
let client_socket = UdpSocket::bind("127.0.0.1:0").unwrap(); | ||
let mut endpoint = quinn::Endpoint::new( | ||
EndpointConfig::default(), | ||
None, | ||
client_socket, | ||
Arc::new(TokioRuntime), | ||
) | ||
.unwrap(); | ||
let default_keypair = Keypair::new(); | ||
endpoint.set_default_client_config(get_client_config( | ||
client_keypair.unwrap_or(&default_keypair), | ||
)); | ||
endpoint | ||
.connect(*addr, "localhost") | ||
.expect("Failed in connecting") | ||
.await | ||
.expect("Failed in waiting") | ||
} |