mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-05-31 05:11:02 +00:00
Finality votes cache (#116)
* removeInMemoryStorage + extract Kovan stuff to runtime * removed comment from the future * remove redundant conversions * remove redundant `u8 as usize` * remove redundant `u8 as usize` * Update modules/ethereum/src/mock.rs Co-authored-by: Hernando Castano <HCastano@users.noreply.github.com> * use hex-literal in kovan config * cargo fmt --all * extracted insert_header * cargo fmt --all * finality cache * cargo fmt --all * cargo fmt --all * impl Default for FinalityVotes Co-authored-by: Hernando Castano <HCastano@users.noreply.github.com> Co-authored-by: Hernando Castano <castano.ha@gmail.com>
This commit is contained in:
committed by
Bastian Köcher
parent
ca8b370de2
commit
643075f7fa
@@ -16,55 +16,108 @@
|
||||
|
||||
use crate::error::Error;
|
||||
use crate::Storage;
|
||||
use primitives::{public_to_address, Address, Header, SealedEmptyStep, H256};
|
||||
use codec::{Decode, Encode};
|
||||
use primitives::{public_to_address, Address, Header, HeaderId, SealedEmptyStep, H256};
|
||||
use sp_io::crypto::secp256k1_ecdsa_recover;
|
||||
use sp_std::prelude::*;
|
||||
use sp_std::{
|
||||
collections::{
|
||||
btree_map::{BTreeMap, Entry},
|
||||
btree_set::BTreeSet,
|
||||
vec_deque::VecDeque,
|
||||
},
|
||||
iter::from_fn,
|
||||
use sp_runtime::RuntimeDebug;
|
||||
use sp_std::collections::{
|
||||
btree_map::{BTreeMap, Entry},
|
||||
btree_set::BTreeSet,
|
||||
vec_deque::VecDeque,
|
||||
};
|
||||
use sp_std::prelude::*;
|
||||
|
||||
/// Cached finality votes for given block.
|
||||
#[derive(RuntimeDebug, Default)]
|
||||
#[cfg_attr(test, derive(PartialEq))]
|
||||
pub struct CachedFinalityVotes<Submitter> {
|
||||
/// Header ancestors that were read while we have been searching for
|
||||
/// cached votes entry. Newest header has index 0.
|
||||
pub unaccounted_ancestry: VecDeque<(HeaderId, Option<Submitter>, Header)>,
|
||||
/// Cached finality votes, if they have been found. The associated
|
||||
/// header is not included into `unaccounted_ancestry`.
|
||||
pub votes: Option<FinalityVotes<Submitter>>,
|
||||
}
|
||||
|
||||
/// Finality effects.
|
||||
#[derive(RuntimeDebug)]
|
||||
#[cfg_attr(test, derive(PartialEq))]
|
||||
pub struct FinalityEffects<Submitter> {
|
||||
/// Finalized headers.
|
||||
pub finalized_headers: Vec<(HeaderId, Option<Submitter>)>,
|
||||
/// Finality votes used in computation.
|
||||
pub votes: FinalityVotes<Submitter>,
|
||||
}
|
||||
|
||||
/// Finality votes for given block.
|
||||
#[derive(RuntimeDebug, Decode, Encode)]
|
||||
#[cfg_attr(test, derive(Clone, PartialEq))]
|
||||
pub struct FinalityVotes<Submitter> {
|
||||
/// Number of votes per each validator.
|
||||
pub votes: BTreeMap<Address, u64>,
|
||||
/// Ancestry blocks with oldest ancestors at the beginning and newest at the
|
||||
/// end of the queue.
|
||||
pub ancestry: VecDeque<FinalityAncestor<Submitter>>,
|
||||
}
|
||||
|
||||
/// Information about block ancestor that is used in computations.
|
||||
#[derive(RuntimeDebug, Decode, Encode)]
|
||||
#[cfg_attr(test, derive(Clone, Default, PartialEq))]
|
||||
pub struct FinalityAncestor<Submitter> {
|
||||
/// Bock id.
|
||||
pub id: HeaderId,
|
||||
/// Block submitter.
|
||||
pub submitter: Option<Submitter>,
|
||||
/// Validators that have signed this block and empty steps on top
|
||||
/// of this block.
|
||||
pub signers: BTreeSet<Address>,
|
||||
}
|
||||
|
||||
/// Tries to finalize blocks when given block is imported.
|
||||
///
|
||||
/// Returns numbers and hashes of finalized blocks in ascending order.
|
||||
pub fn finalize_blocks<S: Storage>(
|
||||
storage: &S,
|
||||
best_finalized_hash: &H256,
|
||||
header_validators: (&H256, &[Address]),
|
||||
hash: &H256,
|
||||
best_finalized: HeaderId,
|
||||
header_validators: (HeaderId, &[Address]),
|
||||
id: HeaderId,
|
||||
submitter: Option<&S::Submitter>,
|
||||
header: &Header,
|
||||
two_thirds_majority_transition: u64,
|
||||
) -> Result<Vec<(u64, H256, Option<S::Submitter>)>, Error> {
|
||||
) -> Result<FinalityEffects<S::Submitter>, Error> {
|
||||
// compute count of voters for every unfinalized block in ancestry
|
||||
let validators = header_validators.1.iter().collect();
|
||||
let (mut votes, mut headers) = prepare_votes(
|
||||
storage,
|
||||
best_finalized_hash,
|
||||
&header_validators.0,
|
||||
let votes = prepare_votes(
|
||||
storage.cached_finality_votes(&header.parent_hash, |hash| {
|
||||
*hash == header_validators.0.hash || *hash == best_finalized.hash
|
||||
}),
|
||||
best_finalized.number,
|
||||
&validators,
|
||||
hash,
|
||||
id,
|
||||
header,
|
||||
submitter,
|
||||
two_thirds_majority_transition,
|
||||
submitter.cloned(),
|
||||
)?;
|
||||
|
||||
// now let's iterate in reverse order && find just finalized blocks
|
||||
let mut newly_finalized = Vec::new();
|
||||
while let Some((oldest_hash, oldest_number, submitter, signers)) = headers.pop_front() {
|
||||
if !is_finalized(&validators, &votes, oldest_number >= two_thirds_majority_transition) {
|
||||
let mut finalized_headers = Vec::new();
|
||||
let mut current_votes = votes.votes.clone();
|
||||
for ancestor in &votes.ancestry {
|
||||
if !is_finalized(
|
||||
&validators,
|
||||
¤t_votes,
|
||||
ancestor.id.number >= two_thirds_majority_transition,
|
||||
) {
|
||||
break;
|
||||
}
|
||||
|
||||
remove_signers_votes(&signers, &mut votes);
|
||||
newly_finalized.push((oldest_number, oldest_hash, submitter));
|
||||
remove_signers_votes(&ancestor.signers, &mut current_votes);
|
||||
finalized_headers.push((ancestor.id, ancestor.submitter.clone()));
|
||||
}
|
||||
|
||||
Ok(newly_finalized)
|
||||
Ok(FinalityEffects {
|
||||
finalized_headers,
|
||||
votes,
|
||||
})
|
||||
}
|
||||
|
||||
/// Returns true if there are enough votes to treat this header as finalized.
|
||||
@@ -78,67 +131,66 @@ fn is_finalized(
|
||||
}
|
||||
|
||||
/// Prepare 'votes' of header and its ancestors' signers.
|
||||
fn prepare_votes<S: Storage>(
|
||||
storage: &S,
|
||||
best_finalized_hash: &H256,
|
||||
validators_begin: &H256,
|
||||
fn prepare_votes<Submitter>(
|
||||
mut cached_votes: CachedFinalityVotes<Submitter>,
|
||||
best_finalized_number: u64,
|
||||
validators: &BTreeSet<&Address>,
|
||||
hash: &H256,
|
||||
id: HeaderId,
|
||||
header: &Header,
|
||||
submitter: Option<&S::Submitter>,
|
||||
two_thirds_majority_transition: u64,
|
||||
) -> Result<
|
||||
(
|
||||
BTreeMap<Address, u64>,
|
||||
VecDeque<(H256, u64, Option<S::Submitter>, BTreeSet<Address>)>,
|
||||
),
|
||||
Error,
|
||||
> {
|
||||
submitter: Option<Submitter>,
|
||||
) -> Result<FinalityVotes<Submitter>, Error> {
|
||||
// this fn can only work with single validators set
|
||||
if !validators.contains(&header.author) {
|
||||
return Err(Error::NotValidator);
|
||||
}
|
||||
|
||||
// prepare iterator of signers of all ancestors of the header
|
||||
// we only take ancestors that are not yet pruned and those signed by
|
||||
// the same set of validators
|
||||
let mut parent_empty_step_signers = empty_steps_signers(header);
|
||||
let ancestry = ancestry(storage, header.parent_hash)
|
||||
.map(|(hash, header, submitter)| {
|
||||
let mut signers = BTreeSet::new();
|
||||
sp_std::mem::swap(&mut signers, &mut parent_empty_step_signers);
|
||||
signers.insert(header.author);
|
||||
// now we have votes that were valid when some block B has been inserted
|
||||
// things may have changed a bit, but we do not need to read anything else
|
||||
// from the db, because we have ancestry
|
||||
// so the only thing we need to do is:
|
||||
// 1) remove votes from blocks that have been finalized after B has been inserted;
|
||||
// 2) add votes from B descendants
|
||||
let mut votes = cached_votes.votes.unwrap_or_default();
|
||||
|
||||
let empty_step_signers = empty_steps_signers(&header);
|
||||
let res = (hash, header.number, submitter, signers);
|
||||
parent_empty_step_signers = empty_step_signers;
|
||||
res
|
||||
})
|
||||
.take_while(|&(hash, _, _, _)| hash != *validators_begin && hash != *best_finalized_hash);
|
||||
|
||||
// now let's iterate built iterator and compute number of validators
|
||||
// 'voted' for each header
|
||||
// we stop when finalized block is met (because we only interested in
|
||||
// just finalized blocks)
|
||||
let mut votes = BTreeMap::new();
|
||||
let mut headers = VecDeque::new();
|
||||
for (hash, number, submitter, signers) in ancestry {
|
||||
add_signers_votes(validators, &signers, &mut votes)?;
|
||||
if is_finalized(validators, &votes, number >= two_thirds_majority_transition) {
|
||||
remove_signers_votes(&signers, &mut votes);
|
||||
// remove votes from finalized blocks
|
||||
while let Some(old_ancestor) = votes.ancestry.pop_front() {
|
||||
if old_ancestor.id.number > best_finalized_number {
|
||||
votes.ancestry.push_front(old_ancestor);
|
||||
break;
|
||||
}
|
||||
|
||||
headers.push_front((hash, number, submitter, signers));
|
||||
remove_signers_votes(&old_ancestor.signers, &mut votes.votes);
|
||||
}
|
||||
|
||||
// update votes with last header vote
|
||||
// add votes from new blocks
|
||||
let mut parent_empty_step_signers = empty_steps_signers(header);
|
||||
let mut unaccounted_ancestry = VecDeque::new();
|
||||
while let Some((ancestor_id, ancestor_submitter, ancestor)) = cached_votes.unaccounted_ancestry.pop_front() {
|
||||
let mut signers = empty_steps_signers(&ancestor);
|
||||
sp_std::mem::swap(&mut signers, &mut parent_empty_step_signers);
|
||||
signers.insert(ancestor.author);
|
||||
|
||||
add_signers_votes(validators, &signers, &mut votes.votes)?;
|
||||
|
||||
unaccounted_ancestry.push_front(FinalityAncestor {
|
||||
id: ancestor_id,
|
||||
submitter: ancestor_submitter,
|
||||
signers,
|
||||
});
|
||||
}
|
||||
votes.ancestry.extend(unaccounted_ancestry);
|
||||
|
||||
// add votes from block itself
|
||||
let mut header_signers = BTreeSet::new();
|
||||
header_signers.insert(header.author);
|
||||
*votes.entry(header.author).or_insert(0) += 1;
|
||||
headers.push_back((*hash, header.number, submitter.cloned(), header_signers));
|
||||
*votes.votes.entry(header.author).or_insert(0) += 1;
|
||||
votes.ancestry.push_back(FinalityAncestor {
|
||||
id,
|
||||
submitter,
|
||||
signers: header_signers,
|
||||
});
|
||||
|
||||
Ok((votes, headers))
|
||||
Ok(votes)
|
||||
}
|
||||
|
||||
/// Increase count of 'votes' for every passed signer.
|
||||
@@ -193,28 +245,21 @@ fn empty_step_signer(empty_step: &SealedEmptyStep, parent_hash: &H256) -> Option
|
||||
.map(|public| public_to_address(&public))
|
||||
}
|
||||
|
||||
/// Return iterator of given header ancestors.
|
||||
pub(crate) fn ancestry<'a, S: Storage>(
|
||||
storage: &'a S,
|
||||
mut parent_hash: H256,
|
||||
) -> impl Iterator<Item = (H256, Header, Option<S::Submitter>)> + 'a {
|
||||
from_fn(move || {
|
||||
let (header, submitter) = storage.header(&parent_hash)?;
|
||||
if header.number == 0 {
|
||||
return None;
|
||||
impl<Submitter> Default for FinalityVotes<Submitter> {
|
||||
fn default() -> Self {
|
||||
FinalityVotes {
|
||||
votes: BTreeMap::new(),
|
||||
ancestry: VecDeque::new(),
|
||||
}
|
||||
|
||||
let hash = parent_hash.clone();
|
||||
parent_hash = header.parent_hash.clone();
|
||||
Some((hash, header, submitter))
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
use crate::mock::{custom_test_ext, genesis, validator, validators_addresses, TestRuntime};
|
||||
use crate::{BridgeStorage, HeaderToImport};
|
||||
use crate::mock::{custom_test_ext, genesis, insert_header, validator, validators_addresses, TestRuntime};
|
||||
use crate::{BridgeStorage, FinalityCache, HeaderToImport};
|
||||
use frame_support::StorageMap;
|
||||
|
||||
#[test]
|
||||
fn verifies_header_author() {
|
||||
@@ -222,9 +267,9 @@ mod tests {
|
||||
assert_eq!(
|
||||
finalize_blocks(
|
||||
&BridgeStorage::<TestRuntime>::new(),
|
||||
&Default::default(),
|
||||
(&Default::default(), &[]),
|
||||
&Default::default(),
|
||||
Default::default(),
|
||||
(Default::default(), &[]),
|
||||
Default::default(),
|
||||
None,
|
||||
&Header::default(),
|
||||
0,
|
||||
@@ -235,7 +280,7 @@ mod tests {
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn prepares_votes() {
|
||||
fn finalize_blocks_works() {
|
||||
custom_test_ext(genesis(), validators_addresses(5)).execute_with(|| {
|
||||
// let's say we have 5 validators (we need 'votes' from 3 validators to achieve
|
||||
// finality)
|
||||
@@ -244,30 +289,32 @@ mod tests {
|
||||
// when header#1 is inserted, nothing is finalized (1 vote)
|
||||
let header1 = Header {
|
||||
author: validator(0).address().as_fixed_bytes().into(),
|
||||
parent_hash: genesis().hash(),
|
||||
parent_hash: genesis().compute_hash(),
|
||||
number: 1,
|
||||
..Default::default()
|
||||
};
|
||||
let hash1 = header1.hash();
|
||||
let id1 = header1.compute_id();
|
||||
let mut header_to_import = HeaderToImport {
|
||||
context: storage.import_context(None, &genesis().hash()).unwrap(),
|
||||
context: storage.import_context(None, &genesis().compute_hash()).unwrap(),
|
||||
is_best: true,
|
||||
hash: hash1,
|
||||
id: id1,
|
||||
header: header1,
|
||||
total_difficulty: 0.into(),
|
||||
enacted_change: None,
|
||||
scheduled_change: None,
|
||||
finality_votes: Default::default(),
|
||||
};
|
||||
assert_eq!(
|
||||
finalize_blocks(
|
||||
&storage,
|
||||
&Default::default(),
|
||||
(&Default::default(), &validators_addresses(5)),
|
||||
&hash1,
|
||||
Default::default(),
|
||||
(Default::default(), &validators_addresses(5)),
|
||||
id1,
|
||||
None,
|
||||
&header_to_import.header,
|
||||
u64::max_value(),
|
||||
),
|
||||
)
|
||||
.map(|eff| eff.finalized_headers),
|
||||
Ok(Vec::new()),
|
||||
);
|
||||
storage.insert_header(header_to_import.clone());
|
||||
@@ -275,22 +322,23 @@ mod tests {
|
||||
// when header#2 is inserted, nothing is finalized (2 votes)
|
||||
header_to_import.header = Header {
|
||||
author: validator(1).address().as_fixed_bytes().into(),
|
||||
parent_hash: hash1,
|
||||
parent_hash: id1.hash,
|
||||
number: 2,
|
||||
..Default::default()
|
||||
};
|
||||
header_to_import.hash = header_to_import.header.hash();
|
||||
let hash2 = header_to_import.header.hash();
|
||||
header_to_import.id = header_to_import.header.compute_id();
|
||||
let id2 = header_to_import.header.compute_id();
|
||||
assert_eq!(
|
||||
finalize_blocks(
|
||||
&storage,
|
||||
&Default::default(),
|
||||
(&Default::default(), &validators_addresses(5)),
|
||||
&hash2,
|
||||
Default::default(),
|
||||
(Default::default(), &validators_addresses(5)),
|
||||
id2,
|
||||
None,
|
||||
&header_to_import.header,
|
||||
u64::max_value(),
|
||||
),
|
||||
)
|
||||
.map(|eff| eff.finalized_headers),
|
||||
Ok(Vec::new()),
|
||||
);
|
||||
storage.insert_header(header_to_import.clone());
|
||||
@@ -298,25 +346,192 @@ mod tests {
|
||||
// when header#3 is inserted, header#1 is finalized (3 votes)
|
||||
header_to_import.header = Header {
|
||||
author: validator(2).address().as_fixed_bytes().into(),
|
||||
parent_hash: hash2,
|
||||
parent_hash: id2.hash,
|
||||
number: 3,
|
||||
..Default::default()
|
||||
};
|
||||
header_to_import.hash = header_to_import.header.hash();
|
||||
let hash3 = header_to_import.header.hash();
|
||||
header_to_import.id = header_to_import.header.compute_id();
|
||||
let id3 = header_to_import.header.compute_id();
|
||||
assert_eq!(
|
||||
finalize_blocks(
|
||||
&storage,
|
||||
&Default::default(),
|
||||
(&Default::default(), &validators_addresses(5)),
|
||||
&hash3,
|
||||
Default::default(),
|
||||
(Default::default(), &validators_addresses(5)),
|
||||
id3,
|
||||
None,
|
||||
&header_to_import.header,
|
||||
u64::max_value(),
|
||||
),
|
||||
Ok(vec![(1, hash1, None)]),
|
||||
)
|
||||
.map(|eff| eff.finalized_headers),
|
||||
Ok(vec![(id1, None)]),
|
||||
);
|
||||
storage.insert_header(header_to_import);
|
||||
});
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn cached_votes_are_updated_with_ancestry() {
|
||||
// we're inserting header#5
|
||||
// cached votes are from header#3
|
||||
// header#4 has finalized header#1 and header#2
|
||||
// => when inserting header#5, we need to:
|
||||
// 1) remove votes from header#1 and header#2
|
||||
// 2) add votes from header#4 and header#5
|
||||
let validators = validators_addresses(5);
|
||||
let headers = (1..6)
|
||||
.map(|number| Header {
|
||||
number: number,
|
||||
author: validators[number as usize - 1],
|
||||
..Default::default()
|
||||
})
|
||||
.collect::<Vec<_>>();
|
||||
let ancestry = headers
|
||||
.iter()
|
||||
.map(|header| FinalityAncestor {
|
||||
id: header.compute_id(),
|
||||
signers: vec![header.author].into_iter().collect(),
|
||||
..Default::default()
|
||||
})
|
||||
.collect::<Vec<_>>();
|
||||
let header5 = headers[4].clone();
|
||||
assert_eq!(
|
||||
prepare_votes::<()>(
|
||||
CachedFinalityVotes {
|
||||
unaccounted_ancestry: vec![(headers[3].compute_id(), None, headers[3].clone()),]
|
||||
.into_iter()
|
||||
.collect(),
|
||||
votes: Some(FinalityVotes {
|
||||
votes: vec![(validators[0], 1), (validators[1], 1), (validators[2], 1),]
|
||||
.into_iter()
|
||||
.collect(),
|
||||
ancestry: ancestry[..3].iter().cloned().collect(),
|
||||
}),
|
||||
},
|
||||
2,
|
||||
&validators.iter().collect(),
|
||||
header5.compute_id(),
|
||||
&header5,
|
||||
None,
|
||||
)
|
||||
.unwrap(),
|
||||
FinalityVotes {
|
||||
votes: vec![(validators[2], 1), (validators[3], 1), (validators[4], 1),]
|
||||
.into_iter()
|
||||
.collect(),
|
||||
ancestry: ancestry[2..].iter().cloned().collect(),
|
||||
},
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn prepare_votes_respects_finality_cache() {
|
||||
let validators_addresses = validators_addresses(5);
|
||||
custom_test_ext(genesis(), validators_addresses.clone()).execute_with(move || {
|
||||
// we need signatures of 3 validators to finalize block
|
||||
let mut storage = BridgeStorage::<TestRuntime>::new();
|
||||
|
||||
// headers 1..3 are signed by validator#0
|
||||
// headers 4..6 are signed by validator#1
|
||||
// headers 7..9 are signed by validator#2
|
||||
let mut hashes = Vec::new();
|
||||
let mut headers = Vec::new();
|
||||
let mut ancestry = Vec::new();
|
||||
let mut parent_hash = genesis().compute_hash();
|
||||
for i in 1..10 {
|
||||
let header = Header {
|
||||
author: validator((i - 1) / 3).address().as_fixed_bytes().into(),
|
||||
parent_hash,
|
||||
number: i as _,
|
||||
..Default::default()
|
||||
};
|
||||
let id = header.compute_id();
|
||||
insert_header(&mut storage, header.clone());
|
||||
hashes.push(id.hash);
|
||||
ancestry.push(FinalityAncestor {
|
||||
id: header.compute_id(),
|
||||
submitter: None,
|
||||
signers: vec![header.author].into_iter().collect(),
|
||||
});
|
||||
headers.push(header);
|
||||
parent_hash = id.hash;
|
||||
}
|
||||
|
||||
// when we're inserting header#7 and last finalized header is 0:
|
||||
// check that votes at #7 are computed correctly without cache
|
||||
let expected_votes_at_7 = FinalityVotes {
|
||||
votes: vec![
|
||||
(validators_addresses[0].clone(), 3),
|
||||
(validators_addresses[1].clone(), 3),
|
||||
(validators_addresses[2].clone(), 1),
|
||||
]
|
||||
.into_iter()
|
||||
.collect(),
|
||||
ancestry: ancestry[..7].iter().cloned().collect(),
|
||||
};
|
||||
let id7 = headers[6].compute_id();
|
||||
assert_eq!(
|
||||
prepare_votes(
|
||||
storage.cached_finality_votes(&hashes.get(5).unwrap(), |_| false,),
|
||||
0,
|
||||
&validators_addresses.iter().collect(),
|
||||
id7,
|
||||
headers.get(6).unwrap(),
|
||||
None,
|
||||
)
|
||||
.unwrap(),
|
||||
expected_votes_at_7,
|
||||
);
|
||||
|
||||
// cached votes at #5
|
||||
let expected_votes_at_5 = FinalityVotes {
|
||||
votes: vec![
|
||||
(validators_addresses[0].clone(), 3),
|
||||
(validators_addresses[1].clone(), 2),
|
||||
]
|
||||
.into_iter()
|
||||
.collect(),
|
||||
ancestry: ancestry[..5].iter().cloned().collect(),
|
||||
};
|
||||
FinalityCache::<TestRuntime>::insert(hashes[4], expected_votes_at_5);
|
||||
|
||||
// when we're inserting header#7 and last finalized header is 0:
|
||||
// check that votes at #7 are computed correctly with cache
|
||||
assert_eq!(
|
||||
prepare_votes(
|
||||
storage.cached_finality_votes(&hashes.get(5).unwrap(), |_| false,),
|
||||
0,
|
||||
&validators_addresses.iter().collect(),
|
||||
id7,
|
||||
headers.get(6).unwrap(),
|
||||
None,
|
||||
)
|
||||
.unwrap(),
|
||||
expected_votes_at_7,
|
||||
);
|
||||
|
||||
// when we're inserting header#7 and last finalized header is 3:
|
||||
// check that votes at #7 are computed correctly with cache
|
||||
let expected_votes_at_7 = FinalityVotes {
|
||||
votes: vec![
|
||||
(validators_addresses[1].clone(), 3),
|
||||
(validators_addresses[2].clone(), 1),
|
||||
]
|
||||
.into_iter()
|
||||
.collect(),
|
||||
ancestry: ancestry[3..7].iter().cloned().collect(),
|
||||
};
|
||||
assert_eq!(
|
||||
prepare_votes(
|
||||
storage.cached_finality_votes(&hashes.get(5).unwrap(), |hash| *hash == hashes[2],),
|
||||
3,
|
||||
&validators_addresses.iter().collect(),
|
||||
id7,
|
||||
headers.get(6).unwrap(),
|
||||
None,
|
||||
)
|
||||
.unwrap(),
|
||||
expected_votes_at_7,
|
||||
);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user