Tweak bootstrap::build_known_snapshot_hashes() (logs and tests) (#26144)

This commit is contained in:
Brooks Prumo 2022-06-30 12:36:13 -05:00 committed by GitHub
parent 6ebd4abf41
commit 649229f7b8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
1 changed files with 147 additions and 94 deletions

View File

@ -755,33 +755,35 @@ where
.any(|hay| needle.0 == hay.0 && needle.1 != hay.1) .any(|hay| needle.0 == hay.0 && needle.1 != hay.1)
} }
'outer: for node in nodes { 'to_next_node: for node in nodes {
// First get the full snapshot hashes for each node and add them as the keys in the // First get the full snapshot hashes for each node and add them as the keys in the
// known snapshot hashes map. // known snapshot hashes map.
let full_snapshot_hashes = get_full_snapshot_hashes_for_node(node); let full_snapshot_hashes = get_full_snapshot_hashes_for_node(node);
for full_snapshot_hash in &full_snapshot_hashes { '_to_next_full_snapshot: for full_snapshot_hash in &full_snapshot_hashes {
// Do not add this snapshot hash if there's already a full snapshot hash with the // Do not add this snapshot hash if there's already a full snapshot hash with the
// same slot but with a _different_ hash. // same slot but with a _different_ hash.
// NOTE: Nodes should not produce snapshots at the same slot with _different_ // NOTE: Nodes should not produce snapshots at the same slot with _different_
// hashes. So if it happens, keep the first and ignore the rest. // hashes. So if it happens, keep the first and ignore the rest.
if !is_any_same_slot_and_different_hash( if is_any_same_slot_and_different_hash(full_snapshot_hash, known_snapshot_hashes.keys())
full_snapshot_hash, {
known_snapshot_hashes.keys(),
) {
// Insert a new full snapshot hash into the known snapshot hashes IFF an entry
// doesn't already exist. This is to ensure we don't overwrite existing
// incremental snapshot hashes that may be present for this full snapshot hash.
known_snapshot_hashes
.entry(*full_snapshot_hash)
.or_default();
} else {
warn!( warn!(
"Ignoring all snapshot hashes from node {} since we've seen a different full snapshot hash with this slot. full snapshot hash: {:?}", "Ignoring all snapshot hashes from node {} since we've seen a different full snapshot hash with this slot.\nfull snapshot hash: {:?}",
node, node,
full_snapshot_hash, full_snapshot_hash,
); );
continue 'outer; debug!(
"known full snapshot hashes: {:#?}",
known_snapshot_hashes.keys(),
);
continue 'to_next_node;
} }
// Insert a new full snapshot hash into the known snapshot hashes IFF an entry
// doesn't already exist. This is to ensure we don't overwrite existing
// incremental snapshot hashes that may be present for this full snapshot hash.
let _ = known_snapshot_hashes
.entry(*full_snapshot_hash)
.or_default();
} }
if incremental_snapshot_fetch { if incremental_snapshot_fetch {
@ -794,34 +796,42 @@ where
// has a full snapshot hash that matches its base snapshot hash. // has a full snapshot hash that matches its base snapshot hash.
if !full_snapshot_hashes.contains(&base_snapshot_hash) { if !full_snapshot_hashes.contains(&base_snapshot_hash) {
warn!( warn!(
"Ignoring all incremental snapshot hashes from node {} since its base snapshot hash does not match any of its full snapshot hashes. base snapshot hash: {:?}, full snapshot hashes: {:?}", "Ignoring all incremental snapshot hashes from node {} since its base snapshot hash does not match any of its full snapshot hashes.\nbase snapshot hash: {:?}\nfull snapshot hashes: {:?}",
node, node,
base_snapshot_hash, base_snapshot_hash,
full_snapshot_hashes full_snapshot_hashes
); );
continue 'outer; continue 'to_next_node;
} }
if let Some(known_incremental_snapshot_hashes) = if let Some(known_incremental_snapshot_hashes) =
known_snapshot_hashes.get_mut(&base_snapshot_hash) known_snapshot_hashes.get_mut(&base_snapshot_hash)
{
'to_next_incremental_snapshot: for incremental_snapshot_hash in
&incremental_snapshot_hashes
{ {
// Do not add this snapshot hash if there's already an incremental snapshot // Do not add this snapshot hash if there's already an incremental snapshot
// hash with the same slot, but with a _different_ hash. // hash with the same slot, but with a _different_ hash.
// NOTE: Nodes should not produce snapshots at the same slot with _different_ // NOTE: Nodes should not produce snapshots at the same slot with _different_
// hashes. So if it happens, keep the first and ignore the rest. // hashes. So if it happens, keep the first and ignore the rest.
for incremental_snapshot_hash in &incremental_snapshot_hashes { if is_any_same_slot_and_different_hash(
if !is_any_same_slot_and_different_hash(
incremental_snapshot_hash, incremental_snapshot_hash,
known_incremental_snapshot_hashes.iter(), known_incremental_snapshot_hashes.iter(),
) { ) {
known_incremental_snapshot_hashes.insert(*incremental_snapshot_hash);
} else {
warn!( warn!(
"Ignoring incremental snapshot hash from node {} since we've seen a different incremental snapshot hash with this slot. incremental snapshot hash: {:?}", "Ignoring incremental snapshot hash from node {} since we've seen a different incremental snapshot hash with this slot.\nbase snapshot hash: {:?}\nincremental snapshot hash: {:?}",
node, node,
base_snapshot_hash,
incremental_snapshot_hash, incremental_snapshot_hash,
); );
debug!(
"known incremental snapshot hashes at this slot: {:#?}",
known_incremental_snapshot_hashes.iter(),
);
continue 'to_next_incremental_snapshot;
} }
known_incremental_snapshot_hashes.insert(*incremental_snapshot_hash);
} }
} else { } else {
// Since incremental snapshots *must* have a valid base (i.e. full) // Since incremental snapshots *must* have a valid base (i.e. full)
@ -834,10 +844,12 @@ where
"There must exist a full snapshot hash already in the known snapshot hashes with the same slot but a different hash!", "There must exist a full snapshot hash already in the known snapshot hashes with the same slot but a different hash!",
); );
debug!( debug!(
"Ignoring incremental snapshot hashes from node {} since we've seen a different base snapshot hash with this slot. base snapshot hash: {:?}", "Ignoring incremental snapshot hashes from node {} since we've seen a different base snapshot hash with this slot.\nbase snapshot hash: {:?}\nknown full snapshot hashes: {:?}",
node, node,
base_snapshot_hash, base_snapshot_hash,
known_snapshot_hashes.keys(),
); );
continue 'to_next_node;
} }
} }
} }
@ -1345,6 +1357,15 @@ mod tests {
), ),
); );
// full and incremental snapshots, with different incremental hashes
oracle.insert(
Pubkey::new_unique(),
(
full_snapshot_hashes1.clone(),
Some((*base_snapshot_hash1, incremental_snapshot_hashes2.clone())),
),
);
// full and incremental snapshots, with different hashes // full and incremental snapshots, with different hashes
oracle.insert( oracle.insert(
Pubkey::new_unique(), Pubkey::new_unique(),
@ -1363,6 +1384,15 @@ mod tests {
), ),
); );
// full and incremental snapshots, with different incremental hashes
oracle.insert(
Pubkey::new_unique(),
(
full_snapshot_hashes2.clone(),
Some((*base_snapshot_hash2, incremental_snapshot_hashes1.clone())),
),
);
// handle duplicates as well // handle duplicates as well
oracle.insert( oracle.insert(
Pubkey::new_unique(), Pubkey::new_unique(),
@ -1372,6 +1402,15 @@ mod tests {
), ),
); );
// handle duplicates as well, with different incremental hashes
oracle.insert(
Pubkey::new_unique(),
(
full_snapshot_hashes1.clone(),
Some((*base_snapshot_hash1, incremental_snapshot_hashes2.clone())),
),
);
// handle duplicates, with different hashes // handle duplicates, with different hashes
oracle.insert( oracle.insert(
Pubkey::new_unique(), Pubkey::new_unique(),
@ -1381,25 +1420,34 @@ mod tests {
), ),
); );
// handle duplicates, with different incremental hashes
oracle.insert(
Pubkey::new_unique(),
(
full_snapshot_hashes2.clone(),
Some((*base_snapshot_hash2, incremental_snapshot_hashes1.clone())),
),
);
let node_to_full_snapshot_hashes = |node| oracle.get(node).unwrap().clone().0; let node_to_full_snapshot_hashes = |node| oracle.get(node).unwrap().clone().0;
let node_to_incremental_snapshot_hashes = |node| oracle.get(node).unwrap().clone().1; let node_to_incremental_snapshot_hashes = |node| oracle.get(node).unwrap().clone().1;
let known_snapshot_hashes_with_incremental = build_known_snapshot_hashes( // With incremental snapshots
{
let known_snapshot_hashes = build_known_snapshot_hashes(
oracle.keys(), oracle.keys(),
node_to_full_snapshot_hashes, node_to_full_snapshot_hashes,
node_to_incremental_snapshot_hashes, node_to_incremental_snapshot_hashes,
true, true,
); );
let mut known_full_snapshot_hashes: Vec<_> = known_snapshot_hashes_with_incremental let mut known_full_snapshot_hashes: Vec<_> =
.keys() known_snapshot_hashes.keys().copied().collect();
.copied()
.collect();
known_full_snapshot_hashes.sort_unstable(); known_full_snapshot_hashes.sort_unstable();
let known_base_snapshot_hash = known_full_snapshot_hashes.last().unwrap(); let known_base_snapshot_hash = known_full_snapshot_hashes.last().unwrap();
let mut known_incremental_snapshot_hashes: Vec<_> = known_snapshot_hashes_with_incremental let mut known_incremental_snapshot_hashes: Vec<_> = known_snapshot_hashes
.get(known_base_snapshot_hash) .get(known_base_snapshot_hash)
.unwrap() .unwrap()
.iter() .iter()
@ -1407,6 +1455,13 @@ mod tests {
.collect(); .collect();
known_incremental_snapshot_hashes.sort_unstable(); known_incremental_snapshot_hashes.sort_unstable();
// The resulting `known_snapshot_hashes` can be different from run-to-run due to how
// `oracle.keys()` returns nodes during iteration. Because of that, we cannot just assert
// the full and incremental snapshot hashes are `full_snapshot_hashes1` and
// `incremental_snapshot_hashes2`. Instead, we assert that the full and incremental
// snapshot hashes are exactly one or the other, since it depends on which nodes are seen
// "first" when building the known snapshot hashes.
assert!( assert!(
known_full_snapshot_hashes == full_snapshot_hashes1 known_full_snapshot_hashes == full_snapshot_hashes1
|| known_full_snapshot_hashes == full_snapshot_hashes2 || known_full_snapshot_hashes == full_snapshot_hashes2
@ -1414,49 +1469,47 @@ mod tests {
if known_full_snapshot_hashes == full_snapshot_hashes1 { if known_full_snapshot_hashes == full_snapshot_hashes1 {
assert_eq!(known_base_snapshot_hash, base_snapshot_hash1); assert_eq!(known_base_snapshot_hash, base_snapshot_hash1);
assert_eq!(
known_incremental_snapshot_hashes,
incremental_snapshot_hashes1
);
} else { } else {
assert_eq!(known_base_snapshot_hash, base_snapshot_hash2); assert_eq!(known_base_snapshot_hash, base_snapshot_hash2);
assert_eq!( }
known_incremental_snapshot_hashes,
incremental_snapshot_hashes2 assert!(
known_incremental_snapshot_hashes == incremental_snapshot_hashes1
|| known_incremental_snapshot_hashes == incremental_snapshot_hashes2
); );
} }
let known_snapshot_hashes_without_incremental = build_known_snapshot_hashes( // Without incremental snapshots
{
let known_snapshot_hashes = build_known_snapshot_hashes(
oracle.keys(), oracle.keys(),
node_to_full_snapshot_hashes, node_to_full_snapshot_hashes,
node_to_incremental_snapshot_hashes, node_to_incremental_snapshot_hashes,
false, false,
); );
let mut known_full_snapshot_hashes: Vec<_> = known_snapshot_hashes_without_incremental let mut known_full_snapshot_hashes: Vec<_> =
.keys() known_snapshot_hashes.keys().copied().collect();
.copied()
.collect();
known_full_snapshot_hashes.sort_unstable(); known_full_snapshot_hashes.sort_unstable();
let known_base_snapshot_hash = known_full_snapshot_hashes.last().unwrap(); let known_base_snapshot_hash = known_full_snapshot_hashes.last().unwrap();
let known_incremental_snapshot_hashes: Vec<_> = known_snapshot_hashes_without_incremental let known_incremental_snapshot_hashes =
.get(known_base_snapshot_hash) known_snapshot_hashes.get(known_base_snapshot_hash).unwrap();
.unwrap()
.iter() // The resulting `known_snapshot_hashes` can be different from run-to-run due to how
.copied() // `oracle.keys()` returns nodes during iteration. Because of that, we cannot just
.collect(); // assert the full snapshot hashes are `full_snapshot_hashes1`. Instead, we assert
// that the full snapshot hashes are exactly one or the other, since it depends on
// which nodes are seen "first" when building the known snapshot hashes.
assert!( assert!(
known_full_snapshot_hashes == full_snapshot_hashes1 known_full_snapshot_hashes == full_snapshot_hashes1
|| known_full_snapshot_hashes == full_snapshot_hashes2 || known_full_snapshot_hashes == full_snapshot_hashes2
); );
assert_eq!( assert!(known_incremental_snapshot_hashes.is_empty());
known_incremental_snapshot_hashes, }
Vec::<(Slot, Hash)>::new()
);
} }
#[test] #[test]