reth_discv5/
enr.rs
1use discv5::enr::{CombinedPublicKey, EnrPublicKey, NodeId};
5use enr::Enr;
6use reth_network_peers::{id2pk, pk2id, PeerId};
7use secp256k1::{PublicKey, SecretKey};
8
9pub fn enr_to_discv4_id(enr: &discv5::Enr) -> Option<PeerId> {
12 let pk = enr.public_key();
13 if !matches!(pk, CombinedPublicKey::Secp256k1(_)) {
14 return None
15 }
16
17 let pk = PublicKey::from_slice(&pk.encode()).unwrap();
18
19 Some(pk2id(&pk))
20}
21
22pub fn discv4_id_to_discv5_id(peer_id: PeerId) -> Result<NodeId, secp256k1::Error> {
24 Ok(id2pk(peer_id)?.into())
25}
26
27pub fn discv4_id_to_multiaddr_id(
29 peer_id: PeerId,
30) -> Result<discv5::libp2p_identity::PeerId, secp256k1::Error> {
31 let pk = id2pk(peer_id)?.encode();
32 let pk: discv5::libp2p_identity::PublicKey =
33 discv5::libp2p_identity::secp256k1::PublicKey::try_from_bytes(&pk).unwrap().into();
34
35 Ok(pk.to_peer_id())
36}
37
38#[derive(Debug, Clone)]
40pub struct EnrCombinedKeyWrapper(pub discv5::Enr);
41
42impl From<Enr<SecretKey>> for EnrCombinedKeyWrapper {
43 fn from(value: Enr<SecretKey>) -> Self {
44 let encoded_enr = alloy_rlp::encode(&value);
45 Self(alloy_rlp::Decodable::decode(&mut &encoded_enr[..]).unwrap())
46 }
47}
48
49impl From<EnrCombinedKeyWrapper> for Enr<SecretKey> {
50 fn from(val: EnrCombinedKeyWrapper) -> Self {
51 let encoded_enr = alloy_rlp::encode(&val.0);
52 alloy_rlp::Decodable::decode(&mut &encoded_enr[..]).unwrap()
53 }
54}
55
56#[cfg(test)]
57mod tests {
58 use super::*;
59 use alloy_rlp::Encodable;
60 use discv5::enr::{CombinedKey, EnrKey};
61 use reth_chainspec::{EthereumHardfork, MAINNET};
62 use reth_network_peers::NodeRecord;
63
64 #[test]
65 fn discv5_discv4_id_conversion() {
66 let discv5_pk = CombinedKey::generate_secp256k1().public();
67 let discv5_peer_id = NodeId::from(discv5_pk.clone());
68
69 let pk = secp256k1::PublicKey::from_slice(&discv5_pk.encode()).unwrap();
71 let discv4_peer_id = pk2id(&pk);
72 let discv5_peer_id_from_discv4_peer_id = discv4_id_to_discv5_id(discv4_peer_id).unwrap();
74
75 assert_eq!(discv5_peer_id, discv5_peer_id_from_discv4_peer_id)
76 }
77
78 #[test]
79 fn conversion_to_node_record_from_enr() {
80 const IP: &str = "::";
81 const TCP_PORT: u16 = 30303;
82 const UDP_PORT: u16 = 9000;
83
84 let key = CombinedKey::generate_secp256k1();
85
86 let mut buf = Vec::new();
87 let fork_id = MAINNET.hardfork_fork_id(EthereumHardfork::Frontier);
88 fork_id.unwrap().encode(&mut buf);
89
90 let enr = Enr::builder()
91 .ip6(IP.parse().unwrap())
92 .udp6(UDP_PORT)
93 .tcp6(TCP_PORT)
94 .build(&key)
95 .unwrap();
96
97 let enr = EnrCombinedKeyWrapper(enr).into();
98 let node_record = NodeRecord::try_from(&enr).unwrap();
99
100 assert_eq!(
101 NodeRecord {
102 address: IP.parse().unwrap(),
103 tcp_port: TCP_PORT,
104 udp_port: UDP_PORT,
105 id: pk2id(&enr.public_key())
106 },
107 node_record
108 )
109 }
110}