mirror of
https://github.com/fluencelabs/rust-libp2p
synced 2025-06-24 15:21:33 +00:00
Implementation of the libp2p perf protocol according to https://github.com/libp2p/specs/pull/478/. //CC @MarcoPolo as the author of the specification. **Don't (yet) expect this to produce reliable performance metrics.** Pull-Request: #3508.
141 lines
5.2 KiB
Rust
141 lines
5.2 KiB
Rust
// Copyright 2023 Protocol Labs.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the "Software"),
|
|
// to deal in the Software without restriction, including without limitation
|
|
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
// and/or sell copies of the Software, and to permit persons to whom the
|
|
// Software is furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
// DEALINGS IN THE SOFTWARE.
|
|
|
|
use anyhow::{bail, Result};
|
|
use clap::Parser;
|
|
use futures::{future::Either, StreamExt};
|
|
use libp2p_core::{muxing::StreamMuxerBox, transport::OrTransport, upgrade, Multiaddr, Transport};
|
|
use libp2p_dns::DnsConfig;
|
|
use libp2p_identity::PeerId;
|
|
use libp2p_perf::client::RunParams;
|
|
use libp2p_swarm::{SwarmBuilder, SwarmEvent};
|
|
use log::info;
|
|
|
|
#[derive(Debug, Parser)]
|
|
#[clap(name = "libp2p perf client")]
|
|
struct Opts {
|
|
#[arg(long)]
|
|
server_address: Multiaddr,
|
|
}
|
|
|
|
#[async_std::main]
|
|
async fn main() -> Result<()> {
|
|
env_logger::Builder::from_env(env_logger::Env::default().default_filter_or("info")).init();
|
|
|
|
let opts = Opts::parse();
|
|
|
|
info!("Initiating performance tests with {}", opts.server_address);
|
|
|
|
// Create a random PeerId
|
|
let local_key = libp2p_identity::Keypair::generate_ed25519();
|
|
let local_peer_id = PeerId::from(local_key.public());
|
|
|
|
let transport = {
|
|
let tcp =
|
|
libp2p_tcp::async_io::Transport::new(libp2p_tcp::Config::default().port_reuse(true))
|
|
.upgrade(upgrade::Version::V1Lazy)
|
|
.authenticate(
|
|
libp2p_noise::NoiseAuthenticated::xx(&local_key)
|
|
.expect("Signing libp2p-noise static DH keypair failed."),
|
|
)
|
|
.multiplex(libp2p_yamux::YamuxConfig::default());
|
|
|
|
let quic = {
|
|
let mut config = libp2p_quic::Config::new(&local_key);
|
|
config.support_draft_29 = true;
|
|
libp2p_quic::async_std::Transport::new(config)
|
|
};
|
|
|
|
let dns = DnsConfig::system(OrTransport::new(quic, tcp))
|
|
.await
|
|
.unwrap();
|
|
|
|
dns.map(|either_output, _| match either_output {
|
|
Either::Left((peer_id, muxer)) => (peer_id, StreamMuxerBox::new(muxer)),
|
|
Either::Right((peer_id, muxer)) => (peer_id, StreamMuxerBox::new(muxer)),
|
|
})
|
|
.boxed()
|
|
};
|
|
|
|
let mut swarm = SwarmBuilder::with_async_std_executor(
|
|
transport,
|
|
libp2p_perf::client::Behaviour::default(),
|
|
local_peer_id,
|
|
)
|
|
.substream_upgrade_protocol_override(upgrade::Version::V1Lazy)
|
|
.build();
|
|
|
|
swarm.dial(opts.server_address.clone()).unwrap();
|
|
let server_peer_id = loop {
|
|
match swarm.next().await.unwrap() {
|
|
SwarmEvent::ConnectionEstablished { peer_id, .. } => break peer_id,
|
|
SwarmEvent::OutgoingConnectionError { peer_id, error } => {
|
|
bail!("Outgoing connection error to {:?}: {:?}", peer_id, error);
|
|
}
|
|
e => panic!("{e:?}"),
|
|
}
|
|
};
|
|
|
|
info!(
|
|
"Connection to {} established. Launching benchmarks.",
|
|
opts.server_address
|
|
);
|
|
|
|
swarm.behaviour_mut().perf(
|
|
server_peer_id,
|
|
RunParams {
|
|
to_send: 10 * 1024 * 1024,
|
|
to_receive: 10 * 1024 * 1024,
|
|
},
|
|
)?;
|
|
|
|
let stats = loop {
|
|
match swarm.next().await.unwrap() {
|
|
SwarmEvent::ConnectionEstablished {
|
|
peer_id, endpoint, ..
|
|
} => {
|
|
info!("Established connection to {:?} via {:?}", peer_id, endpoint);
|
|
}
|
|
SwarmEvent::OutgoingConnectionError { peer_id, error } => {
|
|
info!("Outgoing connection error to {:?}: {:?}", peer_id, error);
|
|
}
|
|
SwarmEvent::Behaviour(libp2p_perf::client::Event { id: _, result }) => break result?,
|
|
e => panic!("{e:?}"),
|
|
}
|
|
};
|
|
|
|
let sent_mebibytes = stats.params.to_send as f64 / 1024.0 / 1024.0;
|
|
let sent_time = (stats.timers.write_done - stats.timers.write_start).as_secs_f64();
|
|
let sent_bandwidth_mebibit_second = (sent_mebibytes * 8.0) / sent_time;
|
|
|
|
let received_mebibytes = stats.params.to_receive as f64 / 1024.0 / 1024.0;
|
|
let receive_time = (stats.timers.read_done - stats.timers.write_done).as_secs_f64();
|
|
let receive_bandwidth_mebibit_second = (received_mebibytes * 8.0) / receive_time;
|
|
|
|
info!(
|
|
"Finished run: Sent {sent_mebibytes:.2} MiB in {sent_time:.2} s with \
|
|
{sent_bandwidth_mebibit_second:.2} MiBit/s and received \
|
|
{received_mebibytes:.2} MiB in {receive_time:.2} s with \
|
|
{receive_bandwidth_mebibit_second:.2} MiBit/s",
|
|
);
|
|
|
|
Ok(())
|
|
}
|