From c6f889f08ce2c1359f98a11ea82bf3851fea4ce3 Mon Sep 17 00:00:00 2001 From: behzad nouri Date: Thu, 8 Aug 2024 21:01:28 +0000 Subject: [PATCH 1/2] checks for duplicate instances using the new ContactInfo (#2506) Working towards deprecating NodeInstance CRDS value, the commit adds check for duplicate instances using the new ContactInfo. (cherry picked from commit 1d825df4e1fce7649ca20bb6f8faba0c87a410e0) --- gossip/src/cluster_info.rs | 21 ++++++++----- gossip/src/contact_info.rs | 63 ++++++++++++++++++++++++++++++++++++++ 2 files changed, 76 insertions(+), 8 deletions(-) diff --git a/gossip/src/cluster_info.rs b/gossip/src/cluster_info.rs index a9a14b5557239a..00e0b4869e1702 100644 --- a/gossip/src/cluster_info.rs +++ b/gossip/src/cluster_info.rs @@ -2495,16 +2495,21 @@ impl ClusterInfo { // Check if there is a duplicate instance of // this node with more recent timestamp. - let instance = self.instance.read().unwrap(); - let check_duplicate_instance = |values: &[CrdsValue]| { - if should_check_duplicate_instance { - for value in values { - if instance.check_duplicate(value) { - return Err(GossipError::DuplicateNodeInstance); - } + let check_duplicate_instance = { + let instance = self.instance.read().unwrap(); + let my_contact_info = self.my_contact_info(); + move |values: &[CrdsValue]| { + if should_check_duplicate_instance + && values.iter().any(|value| { + instance.check_duplicate(value) + || matches!(&value.data, CrdsData::ContactInfo(other) + if my_contact_info.check_duplicate(other)) + }) + { + return Err(GossipError::DuplicateNodeInstance); } + Ok(()) } - Ok(()) }; let mut pings = Vec::new(); let mut rng = rand::thread_rng(); diff --git a/gossip/src/contact_info.rs b/gossip/src/contact_info.rs index e1ef93d243a03c..58d66b168fddd8 100644 --- a/gossip/src/contact_info.rs +++ b/gossip/src/contact_info.rs @@ -434,6 +434,14 @@ impl ContactInfo { node.set_serve_repair_quic((addr, port + 4)).unwrap(); node } + + // Returns true if the other contact-info is a duplicate instance of this + // node, with a more recent `outset` timestamp. + #[inline] + #[must_use] + pub(crate) fn check_duplicate(&self, other: &ContactInfo) -> bool { + self.pubkey == other.pubkey && self.outset < other.outset + } } impl Default for ContactInfo { @@ -1015,4 +1023,59 @@ mod tests { Err(Error::InvalidPort(0)) ); } + + #[test] + fn test_check_duplicate() { + let mut rng = rand::thread_rng(); + let mut node = ContactInfo::new( + Keypair::new().pubkey(), + rng.gen(), // wallclock + rng.gen(), // shred_version + ); + // Same contact-info is not a duplicate instance. + { + let other = node.clone(); + assert!(!node.check_duplicate(&other)); + assert!(!other.check_duplicate(&node)); + } + // Updated socket address is not a duplicate instance. + { + let mut other = node.clone(); + other.set_gossip(new_rand_socket(&mut rng)).unwrap(); + other.set_serve_repair(new_rand_socket(&mut rng)).unwrap(); + assert!(!node.check_duplicate(&other)); + assert!(!other.check_duplicate(&node)); + other.remove_serve_repair(); + assert!(!node.check_duplicate(&other)); + assert!(!other.check_duplicate(&node)); + } + // Updated wallclock is not a duplicate instance. + { + let other = node.clone(); + node.set_wallclock(rng.gen()); + assert!(!node.check_duplicate(&other)); + assert!(!other.check_duplicate(&node)); + } + // Different pubkey is not a duplicate instance. + { + let other = ContactInfo::new( + Keypair::new().pubkey(), + rng.gen(), // wallclock + rng.gen(), // shred_version + ); + assert!(!node.check_duplicate(&other)); + assert!(!other.check_duplicate(&node)); + } + // Same pubkey, more recent outset timestamp is a duplicate instance. + { + let other = ContactInfo::new( + node.pubkey, + rng.gen(), // wallclock + rng.gen(), // shred_version + ); + assert!(node.outset < other.outset); + assert!(node.check_duplicate(&other)); + assert!(!other.check_duplicate(&node)); + } + } } From ad8aed3676a9cbfbe1234793f83ba8572d6788a7 Mon Sep 17 00:00:00 2001 From: behzad nouri Date: Fri, 9 Aug 2024 08:57:20 -0500 Subject: [PATCH 2/2] removes unwarp --- gossip/src/contact_info.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/gossip/src/contact_info.rs b/gossip/src/contact_info.rs index 58d66b168fddd8..4b5e0808cb3985 100644 --- a/gossip/src/contact_info.rs +++ b/gossip/src/contact_info.rs @@ -1041,8 +1041,8 @@ mod tests { // Updated socket address is not a duplicate instance. { let mut other = node.clone(); - other.set_gossip(new_rand_socket(&mut rng)).unwrap(); - other.set_serve_repair(new_rand_socket(&mut rng)).unwrap(); + while other.set_gossip(new_rand_socket(&mut rng)).is_err() {} + while other.set_serve_repair(new_rand_socket(&mut rng)).is_err() {} assert!(!node.check_duplicate(&other)); assert!(!other.check_duplicate(&node)); other.remove_serve_repair();