use futures::future::Either as FutureEither;
use hickory_resolver::system_conf;
use libp2p::core::muxing::StreamMuxerBox;
use libp2p::core::transport::timeout::TransportTimeout;
use libp2p::core::transport::upgrade::Version;
use libp2p::core::transport::{Boxed, MemoryTransport, OrTransport};
use libp2p::dns::{tokio::Transport as TokioDnsConfig, ResolverConfig, ResolverOpts};
use libp2p::quic::tokio::Transport as TokioQuicTransport;
use libp2p::quic::Config as QuicConfig;
use libp2p::relay::client::Transport as ClientTransport;
use libp2p::tcp::{tokio::Transport as TokioTcpTransport, Config as GenTcpConfig};
use libp2p::yamux::Config as YamuxConfig;
use libp2p::{identity, noise};
use libp2p::{PeerId, Transport};
use std::io::{self, ErrorKind};
use std::time::Duration;
pub(crate) type TTransport = Boxed<(PeerId, StreamMuxerBox)>;
#[derive(Debug, Clone, Copy)]
pub struct TransportConfig {
pub timeout: Duration,
pub dns_resolver: Option<DnsResolver>,
pub version: UpgradeVersion,
pub enable_quic: bool,
pub quic_max_idle_timeout: Duration,
pub support_quic_draft_29: bool,
}
impl Default for TransportConfig {
fn default() -> Self {
Self {
enable_quic: true,
support_quic_draft_29: false,
timeout: Duration::from_secs(30),
quic_max_idle_timeout: Duration::from_secs(10),
dns_resolver: None,
version: UpgradeVersion::default(),
}
}
}
#[derive(Default, Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum DnsResolver {
Google,
#[default]
Cloudflare,
Local,
None,
}
impl From<DnsResolver> for (ResolverConfig, ResolverOpts) {
fn from(value: DnsResolver) -> Self {
match value {
DnsResolver::Google => (ResolverConfig::google(), Default::default()),
DnsResolver::Cloudflare => (ResolverConfig::cloudflare(), Default::default()),
DnsResolver::Local => system_conf::read_system_conf().unwrap_or_default(),
DnsResolver::None => (ResolverConfig::new(), Default::default()),
}
}
}
#[derive(Default, Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum UpgradeVersion {
Standard,
#[default]
Lazy,
}
impl From<UpgradeVersion> for Version {
fn from(value: UpgradeVersion) -> Self {
match value {
UpgradeVersion::Standard => Version::V1,
UpgradeVersion::Lazy => Version::V1Lazy,
}
}
}
pub(crate) fn build_transport(
keypair: identity::Keypair,
relay: Option<ClientTransport>,
TransportConfig {
timeout,
dns_resolver,
version,
enable_quic,
support_quic_draft_29,
quic_max_idle_timeout,
..
}: TransportConfig,
) -> io::Result<TTransport> {
let noise_config =
noise::Config::new(&keypair).map_err(|e| io::Error::new(ErrorKind::Other, e))?;
let yamux_config = YamuxConfig::default();
let tcp_config = GenTcpConfig::default().nodelay(true).port_reuse(true);
let transport = TokioTcpTransport::new(tcp_config);
let transport_timeout = TransportTimeout::new(transport, timeout);
let (cfg, opts) = dns_resolver.unwrap_or_default().into();
let transport = TokioDnsConfig::custom(transport_timeout, cfg, opts);
let transport = match relay {
Some(relay) => {
let transport = OrTransport::new(relay, transport);
transport
.upgrade(version.into())
.authenticate(noise_config)
.multiplex(yamux_config)
.timeout(timeout)
.boxed()
}
None => transport
.upgrade(version.into())
.authenticate(noise_config)
.multiplex(yamux_config)
.timeout(timeout)
.boxed(),
};
let transport = match enable_quic {
true => {
let mut quic_config = QuicConfig::new(&keypair);
quic_config.support_draft_29 = support_quic_draft_29;
quic_config.max_idle_timeout = quic_max_idle_timeout.as_millis() as _;
quic_config.keep_alive_interval = quic_max_idle_timeout / 2;
let quic_transport = TokioQuicTransport::new(quic_config);
OrTransport::new(quic_transport, transport)
.map(|either_output, _| match either_output {
FutureEither::Left((peer_id, muxer)) => (peer_id, StreamMuxerBox::new(muxer)),
FutureEither::Right((peer_id, muxer)) => (peer_id, StreamMuxerBox::new(muxer)),
})
.boxed()
}
false => transport,
};
Ok(transport)
}
#[allow(dead_code)]
pub(crate) fn memory_transport(
keypair: &identity::Keypair,
relay: Option<ClientTransport>,
) -> io::Result<TTransport> {
let noise_config =
noise::Config::new(keypair).map_err(|e| io::Error::new(ErrorKind::Other, e))?;
let transport = match relay {
Some(relay) => OrTransport::new(relay, MemoryTransport::default())
.upgrade(Version::V1)
.authenticate(noise_config)
.multiplex(YamuxConfig::default())
.timeout(Duration::from_secs(20))
.boxed(),
None => MemoryTransport::default()
.upgrade(Version::V1)
.authenticate(noise_config)
.multiplex(YamuxConfig::default())
.timeout(Duration::from_secs(20))
.boxed(),
};
Ok(transport)
}