From 25ee7d4c9c9fa09db62453ec26fa7a5845f21823 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jo=C3=A3o=20Oliveira?= Date: Tue, 24 Jan 2023 08:49:59 +0000 Subject: [PATCH] fix(autonat): Skip unparsable multiaddr (#3363) With this commit `libp2p-autonat` no longer discards the whole remote payload in case an addr is unparsable, but instead logs the failure and skips the unparsable multiaddr. See libp2p#3244 for details. --- CHANGELOG.md | 7 +++++++ Cargo.toml | 4 ++-- src/protocol.rs | 50 ++++++++++++++++++++++++++++++++++++++++--------- 3 files changed, 50 insertions(+), 11 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index a39c6b2b14a..48d219b0bdc 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -10,6 +10,13 @@ [PR 3153]: https://github.com/libp2p/rust-libp2p/pull/3153 +# 0.9.1 + +- Skip unparsable multiaddr in `DialRequest::from_bytes`. See [PR 3351]. + +[PR 3351]: https://github.com/libp2p/rust-libp2p/pull/3351 + + # 0.9.0 - Update to `libp2p-core` `v0.38.0`. diff --git a/Cargo.toml b/Cargo.toml index 07e83adb066..2264597622c 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -44,8 +44,8 @@ rustc-args = ["--cfg", "docsrs"] [[example]] name = "client" -path = "examples/autonat_client.rs" +path = "examples/autonat_client.rs" [[example]] name = "server" -path = "examples/autonat_server.rs" +path = "examples/autonat_server.rs" diff --git a/src/protocol.rs b/src/protocol.rs index 08ae415342f..9da7bb5a309 100644 --- a/src/protocol.rs +++ b/src/protocol.rs @@ -134,15 +134,16 @@ impl DialRequest { PeerId::try_from(peer_id) .map_err(|_| io::Error::new(io::ErrorKind::InvalidData, "invalid peer id"))? }; - let addrs = { - let mut maddrs = vec![]; - for addr in addrs.into_iter() { - let maddr = Multiaddr::try_from(addr) - .map_err(|err| io::Error::new(io::ErrorKind::InvalidData, err))?; - maddrs.push(maddr); - } - maddrs - }; + let addrs = addrs + .into_iter() + .filter_map(|a| match Multiaddr::try_from(a) { + Ok(a) => Some(a), + Err(e) => { + log::debug!("Unable to parse multiaddr: {e}"); + None + } + }) + .collect(); Ok(Self { peer_id, addresses: addrs, @@ -333,4 +334,35 @@ mod tests { let response2 = DialResponse::from_bytes(&bytes).unwrap(); assert_eq!(response, response2); } + + #[test] + fn test_skip_unparsable_multiaddr() { + let valid_multiaddr: Multiaddr = "/ip6/2001:db8::/tcp/1234".parse().unwrap(); + let valid_multiaddr_bytes = valid_multiaddr.to_vec(); + + let invalid_multiaddr = { + let a = vec![255; 8]; + assert!(Multiaddr::try_from(a.clone()).is_err()); + a + }; + + let msg = structs_proto::Message { + r#type: Some(structs_proto::message::MessageType::Dial.into()), + dial: Some(structs_proto::message::Dial { + peer: Some(structs_proto::message::PeerInfo { + id: Some(PeerId::random().to_bytes()), + addrs: vec![valid_multiaddr_bytes, invalid_multiaddr], + }), + }), + dial_response: None, + }; + + let mut bytes = Vec::with_capacity(msg.encoded_len()); + msg.encode(&mut bytes) + .expect("Vec provides capacity as needed"); + + let request = DialRequest::from_bytes(&bytes).expect("not to fail"); + + assert_eq!(request.addresses, vec![valid_multiaddr]) + } }