use discv5::enr::{CombinedPublicKey, EnrPublicKey, NodeId};
use enr::Enr;
use reth_network_peers::{id2pk, pk2id, PeerId};
use secp256k1::{PublicKey, SecretKey};
pub fn enr_to_discv4_id(enr: &discv5::Enr) -> Option<PeerId> {
let pk = enr.public_key();
if !matches!(pk, CombinedPublicKey::Secp256k1(_)) {
return None
}
let pk = PublicKey::from_slice(&pk.encode()).unwrap();
Some(pk2id(&pk))
}
pub fn discv4_id_to_discv5_id(peer_id: PeerId) -> Result<NodeId, secp256k1::Error> {
Ok(id2pk(peer_id)?.into())
}
pub fn discv4_id_to_multiaddr_id(
peer_id: PeerId,
) -> Result<discv5::libp2p_identity::PeerId, secp256k1::Error> {
let pk = id2pk(peer_id)?.encode();
let pk: discv5::libp2p_identity::PublicKey =
discv5::libp2p_identity::secp256k1::PublicKey::try_from_bytes(&pk).unwrap().into();
Ok(pk.to_peer_id())
}
#[derive(Debug, Clone)]
pub struct EnrCombinedKeyWrapper(pub discv5::Enr);
impl From<Enr<SecretKey>> for EnrCombinedKeyWrapper {
fn from(value: Enr<SecretKey>) -> Self {
let encoded_enr = alloy_rlp::encode(&value);
Self(alloy_rlp::Decodable::decode(&mut &encoded_enr[..]).unwrap())
}
}
impl From<EnrCombinedKeyWrapper> for Enr<SecretKey> {
fn from(val: EnrCombinedKeyWrapper) -> Self {
let encoded_enr = alloy_rlp::encode(&val.0);
alloy_rlp::Decodable::decode(&mut &encoded_enr[..]).unwrap()
}
}
#[cfg(test)]
mod tests {
use super::*;
use alloy_rlp::Encodable;
use discv5::enr::{CombinedKey, EnrKey};
use reth_chainspec::{EthereumHardfork, MAINNET};
use reth_network_peers::NodeRecord;
#[test]
fn discv5_discv4_id_conversion() {
let discv5_pk = CombinedKey::generate_secp256k1().public();
let discv5_peer_id = NodeId::from(discv5_pk.clone());
let pk = secp256k1::PublicKey::from_slice(&discv5_pk.encode()).unwrap();
let discv4_peer_id = pk2id(&pk);
let discv5_peer_id_from_discv4_peer_id = discv4_id_to_discv5_id(discv4_peer_id).unwrap();
assert_eq!(discv5_peer_id, discv5_peer_id_from_discv4_peer_id)
}
#[test]
fn conversion_to_node_record_from_enr() {
const IP: &str = "::";
const TCP_PORT: u16 = 30303;
const UDP_PORT: u16 = 9000;
let key = CombinedKey::generate_secp256k1();
let mut buf = Vec::new();
let fork_id = MAINNET.hardfork_fork_id(EthereumHardfork::Frontier);
fork_id.unwrap().encode(&mut buf);
let enr = Enr::builder()
.ip6(IP.parse().unwrap())
.udp6(UDP_PORT)
.tcp6(TCP_PORT)
.build(&key)
.unwrap();
let enr = EnrCombinedKeyWrapper(enr).into();
let node_record = NodeRecord::try_from(&enr).unwrap();
assert_eq!(
NodeRecord {
address: IP.parse().unwrap(),
tcp_port: TCP_PORT,
udp_port: UDP_PORT,
id: pk2id(&enr.public_key())
},
node_record
)
}
}