mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-05-07 03:48:01 +00:00
f12e22a62d
* new_rounds() * WIP * test add_vote() * test drop() * learn to spell * go get some coffee * cargo fmt * lump everythings together again
277 lines
6.6 KiB
Rust
277 lines
6.6 KiB
Rust
// This file is part of Substrate.
|
|
|
|
// Copyright (C) 2021 Parity Technologies (UK) Ltd.
|
|
// SPDX-License-Identifier: GPL-3.0-or-later WITH Classpath-exception-2.0
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
use std::{collections::BTreeMap, hash::Hash};
|
|
|
|
use log::{debug, trace};
|
|
|
|
use beefy_primitives::{
|
|
crypto::{Public, Signature},
|
|
ValidatorSet, ValidatorSetId,
|
|
};
|
|
use sp_arithmetic::traits::AtLeast32BitUnsigned;
|
|
use sp_runtime::traits::MaybeDisplay;
|
|
|
|
#[derive(Default)]
|
|
struct RoundTracker {
|
|
votes: Vec<(Public, Signature)>,
|
|
}
|
|
|
|
impl RoundTracker {
|
|
fn add_vote(&mut self, vote: (Public, Signature)) -> bool {
|
|
// this needs to handle equivocations in the future
|
|
if self.votes.contains(&vote) {
|
|
return false
|
|
}
|
|
|
|
self.votes.push(vote);
|
|
true
|
|
}
|
|
|
|
fn is_done(&self, threshold: usize) -> bool {
|
|
self.votes.len() >= threshold
|
|
}
|
|
}
|
|
|
|
fn threshold(authorities: usize) -> usize {
|
|
let faulty = authorities.saturating_sub(1) / 3;
|
|
authorities - faulty
|
|
}
|
|
|
|
pub(crate) struct Rounds<Hash, Number> {
|
|
rounds: BTreeMap<(Hash, Number), RoundTracker>,
|
|
validator_set: ValidatorSet<Public>,
|
|
}
|
|
|
|
impl<H, N> Rounds<H, N>
|
|
where
|
|
H: Ord + Hash,
|
|
N: Ord + AtLeast32BitUnsigned + MaybeDisplay,
|
|
{
|
|
pub(crate) fn new(validator_set: ValidatorSet<Public>) -> Self {
|
|
Rounds { rounds: BTreeMap::new(), validator_set }
|
|
}
|
|
}
|
|
|
|
impl<H, N> Rounds<H, N>
|
|
where
|
|
H: Ord + Hash,
|
|
N: Ord + AtLeast32BitUnsigned + MaybeDisplay,
|
|
{
|
|
pub(crate) fn validator_set_id(&self) -> ValidatorSetId {
|
|
self.validator_set.id
|
|
}
|
|
|
|
pub(crate) fn validators(&self) -> Vec<Public> {
|
|
self.validator_set.validators.clone()
|
|
}
|
|
|
|
pub(crate) fn add_vote(&mut self, round: (H, N), vote: (Public, Signature)) -> bool {
|
|
if self.validator_set.validators.iter().any(|id| vote.0 == *id) {
|
|
self.rounds.entry(round).or_default().add_vote(vote)
|
|
} else {
|
|
false
|
|
}
|
|
}
|
|
|
|
pub(crate) fn is_done(&self, round: &(H, N)) -> bool {
|
|
let done = self
|
|
.rounds
|
|
.get(round)
|
|
.map(|tracker| tracker.is_done(threshold(self.validator_set.validators.len())))
|
|
.unwrap_or(false);
|
|
|
|
debug!(target: "beefy", "🥩 Round #{} done: {}", round.1, done);
|
|
|
|
done
|
|
}
|
|
|
|
pub(crate) fn drop(&mut self, round: &(H, N)) -> Option<Vec<Option<Signature>>> {
|
|
trace!(target: "beefy", "🥩 About to drop round #{}", round.1);
|
|
|
|
let signatures = self.rounds.remove(round)?.votes;
|
|
|
|
Some(
|
|
self.validator_set
|
|
.validators
|
|
.iter()
|
|
.map(|authority_id| {
|
|
signatures.iter().find_map(|(id, sig)| {
|
|
if id == authority_id {
|
|
Some(sig.clone())
|
|
} else {
|
|
None
|
|
}
|
|
})
|
|
})
|
|
.collect(),
|
|
)
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use sc_network_test::Block;
|
|
use sp_core::H256;
|
|
use sp_runtime::traits::NumberFor;
|
|
|
|
use beefy_primitives::{crypto::Public, ValidatorSet};
|
|
|
|
use super::Rounds;
|
|
use crate::keystore::tests::Keyring;
|
|
|
|
#[test]
|
|
fn new_rounds() {
|
|
sp_tracing::try_init_simple();
|
|
|
|
let rounds = Rounds::<H256, NumberFor<Block>>::new(ValidatorSet::<Public>::empty());
|
|
|
|
assert_eq!(0, rounds.validator_set_id());
|
|
assert!(rounds.validators().is_empty());
|
|
|
|
let validators = ValidatorSet::<Public> {
|
|
validators: vec![
|
|
Keyring::Alice.public(),
|
|
Keyring::Bob.public(),
|
|
Keyring::Charlie.public(),
|
|
],
|
|
id: 42,
|
|
};
|
|
|
|
let rounds = Rounds::<H256, NumberFor<Block>>::new(validators);
|
|
|
|
assert_eq!(42, rounds.validator_set_id());
|
|
|
|
assert_eq!(
|
|
vec![Keyring::Alice.public(), Keyring::Bob.public(), Keyring::Charlie.public()],
|
|
rounds.validators()
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn add_vote() {
|
|
sp_tracing::try_init_simple();
|
|
|
|
let validators = ValidatorSet::<Public> {
|
|
validators: vec![
|
|
Keyring::Alice.public(),
|
|
Keyring::Bob.public(),
|
|
Keyring::Charlie.public(),
|
|
],
|
|
id: Default::default(),
|
|
};
|
|
|
|
let mut rounds = Rounds::<H256, NumberFor<Block>>::new(validators);
|
|
|
|
assert!(rounds.add_vote(
|
|
(H256::from_low_u64_le(1), 1),
|
|
(Keyring::Alice.public(), Keyring::Alice.sign(b"I am committed"))
|
|
));
|
|
|
|
assert!(!rounds.is_done(&(H256::from_low_u64_le(1), 1)));
|
|
|
|
// invalid vote
|
|
assert!(!rounds.add_vote(
|
|
(H256::from_low_u64_le(1), 1),
|
|
(Keyring::Dave.public(), Keyring::Dave.sign(b"I am committed"))
|
|
));
|
|
|
|
assert!(!rounds.is_done(&(H256::from_low_u64_le(1), 1)));
|
|
|
|
assert!(rounds.add_vote(
|
|
(H256::from_low_u64_le(1), 1),
|
|
(Keyring::Bob.public(), Keyring::Bob.sign(b"I am committed"))
|
|
));
|
|
|
|
assert!(!rounds.is_done(&(H256::from_low_u64_le(1), 1)));
|
|
|
|
assert!(rounds.add_vote(
|
|
(H256::from_low_u64_le(1), 1),
|
|
(Keyring::Charlie.public(), Keyring::Charlie.sign(b"I am committed"))
|
|
));
|
|
|
|
assert!(rounds.is_done(&(H256::from_low_u64_le(1), 1)));
|
|
}
|
|
|
|
#[test]
|
|
fn drop() {
|
|
sp_tracing::try_init_simple();
|
|
|
|
let validators = ValidatorSet::<Public> {
|
|
validators: vec![
|
|
Keyring::Alice.public(),
|
|
Keyring::Bob.public(),
|
|
Keyring::Charlie.public(),
|
|
],
|
|
id: Default::default(),
|
|
};
|
|
|
|
let mut rounds = Rounds::<H256, NumberFor<Block>>::new(validators);
|
|
|
|
// round 1
|
|
rounds.add_vote(
|
|
(H256::from_low_u64_le(1), 1),
|
|
(Keyring::Alice.public(), Keyring::Alice.sign(b"I am committed")),
|
|
);
|
|
rounds.add_vote(
|
|
(H256::from_low_u64_le(1), 1),
|
|
(Keyring::Bob.public(), Keyring::Bob.sign(b"I am committed")),
|
|
);
|
|
|
|
// round 2
|
|
rounds.add_vote(
|
|
(H256::from_low_u64_le(2), 2),
|
|
(Keyring::Alice.public(), Keyring::Alice.sign(b"I am again committed")),
|
|
);
|
|
rounds.add_vote(
|
|
(H256::from_low_u64_le(2), 2),
|
|
(Keyring::Bob.public(), Keyring::Bob.sign(b"I am again committed")),
|
|
);
|
|
|
|
// round 3
|
|
rounds.add_vote(
|
|
(H256::from_low_u64_le(3), 3),
|
|
(Keyring::Alice.public(), Keyring::Alice.sign(b"I am still committed")),
|
|
);
|
|
rounds.add_vote(
|
|
(H256::from_low_u64_le(3), 3),
|
|
(Keyring::Bob.public(), Keyring::Bob.sign(b"I am still committed")),
|
|
);
|
|
|
|
assert_eq!(3, rounds.rounds.len());
|
|
|
|
// drop unknown round
|
|
assert!(rounds.drop(&(H256::from_low_u64_le(5), 5)).is_none());
|
|
assert_eq!(3, rounds.rounds.len());
|
|
|
|
// drop round 2
|
|
let signatures = rounds.drop(&(H256::from_low_u64_le(2), 2)).unwrap();
|
|
|
|
assert_eq!(2, rounds.rounds.len());
|
|
|
|
assert_eq!(
|
|
signatures,
|
|
vec![
|
|
Some(Keyring::Alice.sign(b"I am again committed")),
|
|
Some(Keyring::Bob.sign(b"I am again committed")),
|
|
None
|
|
]
|
|
);
|
|
}
|
|
}
|