diff --git a/polkadot/Cargo.lock b/polkadot/Cargo.lock index 22c14cc8b38ad053778c2c33ac08ee93d40a92bb..61f29625afcd58b920beed0ca3db83a49a34d8c0 100644 --- a/polkadot/Cargo.lock +++ b/polkadot/Cargo.lock @@ -4358,6 +4358,7 @@ dependencies = [ "polkadot-erasure-coding", "polkadot-primitives", "polkadot-validation", + "rand 0.7.3", "sc-network", "sc-network-gossip", "sp-api", diff --git a/polkadot/network/Cargo.toml b/polkadot/network/Cargo.toml index 804cdcb3f11b3bb97a35d8be9632adf58073a5dc..bfee6f34550bdc1afed4689307a778f3036ff16c 100644 --- a/polkadot/network/Cargo.toml +++ b/polkadot/network/Cargo.toml @@ -26,6 +26,7 @@ futures-timer = "2.0" sp-blockchain = { git = "https://github.com/paritytech/substrate", branch = "master" } sp-api = { git = "https://github.com/paritytech/substrate", branch = "master" } wasm-timer = "0.2.4" +rand = "0.7.3" [dev-dependencies] sp-keyring = { git = "https://github.com/paritytech/substrate", branch = "master" } diff --git a/polkadot/network/src/legacy/local_collations.rs b/polkadot/network/src/legacy/local_collations.rs index f1a6615e88b82cb044de6823be6b44d03f2363c3..6bc9b985a7c1a81b2fe0b5edc764015b808546c5 100644 --- a/polkadot/network/src/legacy/local_collations.rs +++ b/polkadot/network/src/legacy/local_collations.rs @@ -24,6 +24,7 @@ use crate::legacy::collator_pool::Role; use std::collections::{HashMap, HashSet}; use std::time::Duration; use wasm_timer::Instant; +use rand::seq::SliceRandom; const LIVE_FOR: Duration = Duration::from_secs(60 * 5); @@ -106,9 +107,7 @@ impl<C: Clone> LocalCollations<C> { relay_parent: Hash, targets: HashSet<ValidatorId>, collation: C - ) - -> impl Iterator<Item=(ValidatorId, C)> + 'a - { + ) -> impl Iterator<Item=(ValidatorId, C)> + 'a { self.local_collations.insert(relay_parent, LocalCollation { targets, collation, @@ -119,8 +118,17 @@ impl<C: Clone> LocalCollations<C> { .expect("just inserted to this key; qed"); let borrowed_collation = &local.collation; + + // If we are conected to multiple validators, + // make sure we always send the collation to one of the validators + // we are registered as backup. This ensures that one collator that + // is primary at multiple validators, desn't block the Parachain from progressing. + let mut rng = rand::thread_rng(); + let diff = local.targets.difference(&self.primary_for).collect::<Vec<_>>(); + local.targets .intersection(&self.primary_for) + .chain(diff.choose(&mut rng).map(|r| r.clone())) .map(move |k| (k.clone(), borrowed_collation.clone())) } @@ -149,7 +157,7 @@ mod tests { }; let mut tracker = LocalCollations::new(); - assert!(tracker.add_collation(relay_parent, targets, 5).next().is_none()); + assert!(tracker.add_collation(relay_parent, targets, 5).next().is_some()); assert_eq!(tracker.note_validator_role(key, Role::Primary), vec![(relay_parent, 5)]); } @@ -165,7 +173,7 @@ mod tests { }; let mut tracker: LocalCollations<u8> = LocalCollations::new(); - assert!(tracker.add_collation(relay_parent, targets, 5).next().is_none()); + assert!(tracker.add_collation(relay_parent, targets, 5).next().is_some()); assert!(tracker.note_validator_role(orig_key.clone(), Role::Primary).is_empty()); assert_eq!(tracker.fresh_key(&orig_key, &new_key), vec![(relay_parent, 5u8)]); }