Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix(quic): allow listening on ipv4 and ipv6 separately #4289

Merged
merged 11 commits into from
Aug 8, 2023
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

7 changes: 7 additions & 0 deletions transports/quic/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,10 @@
## 0.9.1-alpha - unreleased

- Allow listening on ipv4 and ipv6 seperately.
thomaseizinger marked this conversation as resolved.
Show resolved Hide resolved
See [PR 4289].

[PR 4289]: https://github.com/libp2p/rust-libp2p/pull/4289

## 0.9.0-alpha

- Use `quinn` instead of `quinn-proto`.
Expand Down
1 change: 1 addition & 0 deletions transports/quic/Cargo.toml
mxinden marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ rand = "0.8.5"
rustls = { version = "0.21.2", default-features = false }
thiserror = "1.0.44"
tokio = { version = "1.29.1", default-features = false, features = ["net", "rt", "time"], optional = true }
socket2 = { version = "0.5.3" }
mxinden marked this conversation as resolved.
Show resolved Hide resolved

[features]
tokio = ["dep:tokio", "if-watch/tokio", "quinn/runtime-tokio"]
Expand Down
43 changes: 41 additions & 2 deletions transports/quic/src/transport.rs
Original file line number Diff line number Diff line change
Expand Up @@ -37,12 +37,13 @@ use libp2p_core::{
Transport,
};
use libp2p_identity::PeerId;
use socket2::{Domain, Socket, Type};
use std::collections::hash_map::{DefaultHasher, Entry};
use std::collections::HashMap;
use std::fmt;
use std::hash::{Hash, Hasher};
use std::net::{IpAddr, Ipv4Addr, Ipv6Addr, UdpSocket};
use std::time::Duration;
use std::{fmt, io};
use std::{
net::SocketAddr,
pin::Pin,
Expand Down Expand Up @@ -172,6 +173,22 @@ impl<P: Provider> GenTransport<P> {
}
}
}

fn create_socket(&self, socket_addr: SocketAddr) -> io::Result<UdpSocket> {
let domain = if socket_addr.is_ipv4() {
Domain::IPV4
} else {
Domain::IPV6
};
let socket = Socket::new(domain, Type::DGRAM, Some(socket2::Protocol::UDP))?;
thomaseizinger marked this conversation as resolved.
Show resolved Hide resolved
mxinden marked this conversation as resolved.
Show resolved Hide resolved
if socket_addr.is_ipv6() {
socket.set_only_v6(true)?;
}

socket.bind(&socket_addr.into())?;

Ok(socket.into())
}
}

impl<P: Provider> Transport for GenTransport<P> {
Expand All @@ -188,7 +205,8 @@ impl<P: Provider> Transport for GenTransport<P> {
let (socket_addr, version, _peer_id) = self.remote_multiaddr_to_socketaddr(addr, false)?;
let endpoint_config = self.quinn_config.endpoint_config.clone();
let server_config = self.quinn_config.server_config.clone();
let socket = UdpSocket::bind(socket_addr).map_err(Self::Error::from)?;
let socket = self.create_socket(socket_addr).map_err(Self::Error::from)?;

let socket_c = socket.try_clone().map_err(Self::Error::from)?;
let endpoint = Self::new_endpoint(endpoint_config, Some(server_config), socket)?;
let listener = Listener::new(
Expand Down Expand Up @@ -888,4 +906,25 @@ mod test {
.unwrap();
assert!(!transport.dialer.contains_key(&SocketFamily::Ipv4));
}

#[cfg(feature = "tokio")]
#[tokio::test]
async fn test_listens_ipv4_ipv6_separately() {
mxinden marked this conversation as resolved.
Show resolved Hide resolved
let keypair = libp2p_identity::Keypair::generate_ed25519();
let config = Config::new(&keypair);
let mut transport = crate::tokio::Transport::new(config);

transport
.listen_on(
ListenerId::next(),
"/ip4/0.0.0.0/udp/4001/quic-v1".parse().unwrap(),
)
.unwrap();
transport
.listen_on(
ListenerId::next(),
"/ip6/::/udp/4001/quic-v1".parse().unwrap(),
)
.unwrap();
}
}