fmt + clippy + fix tests

This commit is contained in:
Boog900 2023-12-17 03:13:30 +00:00
parent b0588fad2b
commit 376a41deb2
No known key found for this signature in database
GPG key ID: 5401367FB7302004
10 changed files with 101 additions and 106 deletions

View file

@ -65,6 +65,8 @@ syn = "2.0.37"
[dev-dependencies] [dev-dependencies]
monero-consensus = {path = "./rules", features = ["proptest"]}
tokio = {version = "1", features = ["rt-multi-thread", "macros"]} tokio = {version = "1", features = ["rt-multi-thread", "macros"]}
proptest = "1" proptest = "1"
proptest-derive = "0.4.0" proptest-derive = "0.4.0"

View file

@ -23,7 +23,10 @@ thiserror = { workspace = true }
rayon = { workspace = true, optional = true } rayon = { workspace = true, optional = true }
# Testing
proptest = {workspace = true, optional = true} proptest = {workspace = true, optional = true}
proptest-derive = {workspace = true, optional = true} proptest-derive = {workspace = true, optional = true}
[dev-dependencies]
proptest = {workspace = true}
proptest-derive = {workspace = true}
tokio = {version = "1.35.0", features = ["rt-multi-thread", "macros"]}

View file

@ -4,7 +4,7 @@
//! an identifier for every current hard-fork. //! an identifier for every current hard-fork.
//! //!
//! This module also contains a [`HFVotes`] struct which keeps track of current blockchain voting, and //! This module also contains a [`HFVotes`] struct which keeps track of current blockchain voting, and
//! has a method [`HFVotes::check_next_hard_fork`] to check if the next hard-fork should be activated. //! has a method [`HFVotes::current_fork`] to check if the next hard-fork should be activated.
//! //!
use monero_serai::block::BlockHeader; use monero_serai::block::BlockHeader;
use std::{ use std::{
@ -13,12 +13,15 @@ use std::{
time::Duration, time::Duration,
}; };
#[cfg(test)]
mod tests;
/// Target block time for hf 1. /// Target block time for hf 1.
const BLOCK_TIME_V1: Duration = Duration::from_secs(60); const BLOCK_TIME_V1: Duration = Duration::from_secs(60);
/// Target block time from v2. /// Target block time from v2.
const BLOCK_TIME_V2: Duration = Duration::from_secs(120); const BLOCK_TIME_V2: Duration = Duration::from_secs(120);
const NUMB_OF_HARD_FORKS: usize = 16; pub const NUMB_OF_HARD_FORKS: usize = 16;
#[derive(Debug, Copy, Clone, PartialEq, Eq, thiserror::Error)] #[derive(Debug, Copy, Clone, PartialEq, Eq, thiserror::Error)]
pub enum HardForkError { pub enum HardForkError {
@ -51,6 +54,10 @@ impl HFsInfo {
self.0[*hf as usize - 1] self.0[*hf as usize - 1]
} }
pub const fn new(hfs: [HFInfo; NUMB_OF_HARD_FORKS]) -> Self {
Self(hfs)
}
/// Returns the main-net hard-fork information. /// Returns the main-net hard-fork information.
/// ///
/// https://cuprate.github.io/monero-book/consensus_rules/hardforks.html#Mainnet-Hard-Forks /// https://cuprate.github.io/monero-book/consensus_rules/hardforks.html#Mainnet-Hard-Forks
@ -78,7 +85,7 @@ impl HFsInfo {
/// An identifier for every hard-fork Monero has had. /// An identifier for every hard-fork Monero has had.
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Copy, Clone)] #[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Copy, Clone)]
#[cfg_attr(proptest, derive(proptest_derive::Arbitrary))] #[cfg_attr(any(feature = "proptest", test), derive(proptest_derive::Arbitrary))]
#[repr(u8)] #[repr(u8)]
pub enum HardFork { pub enum HardFork {
V1 = 1, V1 = 1,
@ -245,29 +252,30 @@ impl HFVotes {
/// activated. /// activated.
/// ///
/// https://cuprate.github.io/monero-book/consensus_rules/hardforks.html#accepting-a-fork /// https://cuprate.github.io/monero-book/consensus_rules/hardforks.html#accepting-a-fork
pub fn check_next_hard_fork( pub fn current_fork(
&self, &self,
current_hf: &HardFork, current_hf: &HardFork,
current_height: u64, current_height: u64,
window: u64, window: u64,
hfs_info: &HFsInfo, hfs_info: &HFsInfo,
) -> Option<HardFork> { ) -> HardFork {
let mut approved_next_hf = None; let mut current_hf = *current_hf;
while let Some(next_hf) = current_hf.next_fork() { while let Some(next_hf) = current_hf.next_fork() {
let hf_info = hfs_info.info_for_hf(&next_hf); let hf_info = hfs_info.info_for_hf(&next_hf);
if current_height >= hf_info.height if current_height >= hf_info.height
&& self.votes_for_hf(&next_hf) >= votes_needed(hf_info.threshold, window) && self.votes_for_hf(&next_hf) >= votes_needed(hf_info.threshold, window)
{ {
approved_next_hf = Some(next_hf); current_hf = next_hf;
} else { } else {
// if we don't have enough votes for this fork any future fork won't have enough votes // if we don't have enough votes for this fork any future fork won't have enough votes
// as votes are cumulative. // as votes are cumulative.
// TODO: If a future fork has a lower threshold that could not be true, but as all current forks // TODO: If a future fork has a lower threshold that could not be true, but as all current forks
// have threshold 0 it is ok for now. // have threshold 0 it is ok for now.
return approved_next_hf; return current_hf;
} }
} }
approved_next_hf current_hf
} }
} }

View file

@ -0,0 +1,69 @@
use std::convert::TryInto;
use proptest::{arbitrary::any, prop_assert_eq, prop_compose, proptest};
use crate::hard_forks::{HFVotes, HardFork, NUMB_OF_HARD_FORKS};
const TEST_WINDOW_SIZE: u64 = 25;
#[test]
fn next_hard_forks() {
let mut prev = HardFork::V1;
let mut next = HardFork::V2;
for _ in 2..NUMB_OF_HARD_FORKS {
assert!(prev < next);
prev = next;
next = next.next_fork().unwrap();
}
}
#[test]
fn hard_forks_defined() {
for fork in 1..=NUMB_OF_HARD_FORKS {
HardFork::from_version(fork.try_into().unwrap()).unwrap();
}
}
prop_compose! {
/// Generates an arbitrary full [`HFVotes`].
fn arb_full_hf_votes()
(
// we can't use HardFork as for some reason it overflows the stack, so we use u8.
votes in any::<[u8; TEST_WINDOW_SIZE as usize]>()
) -> HFVotes {
let mut vote_count = HFVotes::new(TEST_WINDOW_SIZE as usize);
for vote in votes {
vote_count.add_vote_for_hf(&HardFork::from_vote(vote % 17));
}
vote_count
}
}
proptest! {
#[test]
fn hf_vote_counter_total_correct(hf_votes in arb_full_hf_votes()) {
prop_assert_eq!(hf_votes.total_votes(), u64::try_from(hf_votes.vote_list.len()).unwrap());
let mut votes = [0_u64; NUMB_OF_HARD_FORKS];
for vote in hf_votes.vote_list.iter() {
// manually go through the list of votes tallying
votes[*vote as usize - 1] += 1;
}
prop_assert_eq!(votes, hf_votes.votes);
}
#[test]
fn window_size_kept_constant(mut hf_votes in arb_full_hf_votes(), new_votes in any::<Vec<HardFork>>()) {
for new_vote in new_votes.into_iter() {
hf_votes.add_vote_for_hf(&new_vote);
prop_assert_eq!(hf_votes.total_votes(), TEST_WINDOW_SIZE)
}
}
#[test]
fn votes_out_of_range(high_vote in (NUMB_OF_HARD_FORKS+ 1).try_into().unwrap()..u8::MAX) {
prop_assert_eq!(HardFork::from_vote(0), HardFork::V1);
prop_assert_eq!(HardFork::from_vote(u8::try_from(NUMB_OF_HARD_FORKS).unwrap() + 1_u8), HardFork::from_vote(high_vote));
}
}

View file

@ -3,7 +3,7 @@ use std::time::{SystemTime, UNIX_EPOCH};
pub mod blocks; pub mod blocks;
mod decomposed_amount; mod decomposed_amount;
pub mod genesis; pub mod genesis;
mod hard_forks; pub mod hard_forks;
pub mod miner_tx; pub mod miner_tx;
pub mod signatures; pub mod signatures;
pub mod transactions; pub mod transactions;

View file

@ -171,7 +171,7 @@ where
}); });
while let Some(incoming_blocks) = incoming_blocks.next().await { while let Some(incoming_blocks) = incoming_blocks.next().await {
let mut height = 0; let mut height;
for block in incoming_blocks { for block in incoming_blocks {
let VerifyBlockResponse::MainChain(verified_block_info) = block_verifier let VerifyBlockResponse::MainChain(verified_block_info) = block_verifier
.ready() .ready()

View file

@ -6,7 +6,7 @@ use std::{
}; };
use futures::FutureExt; use futures::FutureExt;
use monero_serai::{block::Block, transaction::Input}; use monero_serai::block::Block;
use tower::{Service, ServiceExt}; use tower::{Service, ServiceExt};
use monero_consensus::{ use monero_consensus::{

View file

@ -119,14 +119,12 @@ impl HardForkState {
/// ///
/// https://cuprate.github.io/monero-docs/consensus_rules/hardforks.html#accepting-a-fork /// https://cuprate.github.io/monero-docs/consensus_rules/hardforks.html#accepting-a-fork
fn check_set_new_hf(&mut self) { fn check_set_new_hf(&mut self) {
if let Some(next_fork) = self.votes.check_next_hard_fork( self.current_hardfork = self.votes.current_fork(
&self.current_hardfork, &self.current_hardfork,
self.last_height + 1, self.last_height + 1,
self.config.window, self.config.window,
&self.config.info, &self.config.info,
) { );
self.current_hardfork = next_fork;
}
} }
pub fn current_hardfork(&self) -> HardFork { pub fn current_hardfork(&self) -> HardFork {
@ -134,13 +132,6 @@ impl HardForkState {
} }
} }
/// Returns the votes needed for this fork.
///
/// https://cuprate.github.io/monero-docs/consensus_rules/hardforks.html#accepting-a-fork
pub fn votes_needed(threshold: u64, window: u64) -> u64 {
(threshold * window + 99) / 100
}
#[instrument(name = "get_votes", skip(database))] #[instrument(name = "get_votes", skip(database))]
async fn get_votes_in_range<D: Database>( async fn get_votes_in_range<D: Database>(
database: D, database: D,

View file

@ -1,8 +1,7 @@
use std::convert::TryInto; use monero_consensus::hard_forks::{HFInfo, HardFork, NUMB_OF_HARD_FORKS};
use monero_consensus::HFsInfo;
use proptest::{arbitrary::any, prop_assert_eq, prop_compose, proptest}; use super::{HardForkConfig, HardForkState};
use super::{HFInfo, HFVotes, HardFork, HardForkConfig, HardForkState, NUMB_OF_HARD_FORKS};
use crate::test_utils::mock_db::*; use crate::test_utils::mock_db::*;
const TEST_WINDOW_SIZE: u64 = 25; const TEST_WINDOW_SIZE: u64 = 25;
@ -28,27 +27,9 @@ const TEST_HFS: [HFInfo; NUMB_OF_HARD_FORKS] = [
pub const TEST_HARD_FORK_CONFIG: HardForkConfig = HardForkConfig { pub const TEST_HARD_FORK_CONFIG: HardForkConfig = HardForkConfig {
window: TEST_WINDOW_SIZE, window: TEST_WINDOW_SIZE,
forks: TEST_HFS, info: HFsInfo::new(TEST_HFS),
}; };
#[test]
fn next_hard_forks() {
let mut prev = HardFork::V1;
let mut next = HardFork::V2;
for _ in 2..NUMB_OF_HARD_FORKS {
assert!(prev < next);
prev = next;
next = next.next_fork().unwrap();
}
}
#[test]
fn hard_forks_defined() {
for fork in 1..=NUMB_OF_HARD_FORKS {
HardFork::from_version(&fork.try_into().unwrap()).unwrap();
}
}
#[tokio::test] #[tokio::test]
async fn hard_fork_set_depends_on_top_block() { async fn hard_fork_set_depends_on_top_block() {
let mut db_builder = DummyDatabaseBuilder::default(); let mut db_builder = DummyDatabaseBuilder::default();
@ -72,47 +53,3 @@ async fn hard_fork_set_depends_on_top_block() {
assert_eq!(state.current_hardfork, HardFork::V14); assert_eq!(state.current_hardfork, HardFork::V14);
} }
prop_compose! {
/// Generates an arbitrary full [`HFVotes`].
fn arb_full_hf_votes()
(
// we can't use HardFork as for some reason it overflows the stack, so we use u8.
votes in any::<[u8; TEST_WINDOW_SIZE as usize]>()
) -> HFVotes {
let mut vote_count = HFVotes::new(TEST_WINDOW_SIZE as usize);
for vote in votes {
vote_count.add_vote_for_hf(&HardFork::from_vote(&(vote % 17)));
}
vote_count
}
}
proptest! {
#[test]
fn hf_vote_counter_total_correct(hf_votes in arb_full_hf_votes()) {
prop_assert_eq!(hf_votes.total_votes(), u64::try_from(hf_votes.vote_list.len()).unwrap());
let mut votes = [0_u64; NUMB_OF_HARD_FORKS];
for vote in hf_votes.vote_list.iter() {
// manually go through the list of votes tallying
votes[*vote as usize - 1] += 1;
}
prop_assert_eq!(votes, hf_votes.votes);
}
#[test]
fn window_size_kept_constant(mut hf_votes in arb_full_hf_votes(), new_votes in any::<Vec<HardFork>>()) {
for new_vote in new_votes.into_iter() {
hf_votes.add_vote_for_hf(&new_vote);
prop_assert_eq!(hf_votes.total_votes(), TEST_WINDOW_SIZE)
}
}
#[test]
fn votes_out_of_range(high_vote in (NUMB_OF_HARD_FORKS+ 1).try_into().unwrap()..u8::MAX) {
prop_assert_eq!(HardFork::from_vote(&0), HardFork::V1);
prop_assert_eq!(HardFork::from_vote(&NUMB_OF_HARD_FORKS.try_into().unwrap()), HardFork::from_vote(&high_vote));
}
}

View file

@ -3,8 +3,6 @@ use std::{
time::{SystemTime, UNIX_EPOCH}, time::{SystemTime, UNIX_EPOCH},
}; };
use curve25519_dalek::edwards::CompressedEdwardsY;
/// Spawns a task for the rayon thread pool and awaits the result without blocking the async runtime. /// Spawns a task for the rayon thread pool and awaits the result without blocking the async runtime.
pub(crate) async fn rayon_spawn_async<F, R>(f: F) -> R pub(crate) async fn rayon_spawn_async<F, R>(f: F) -> R
where where
@ -54,16 +52,3 @@ pub(crate) fn current_time() -> u64 {
.unwrap() .unwrap()
.as_secs() .as_secs()
} }
/// Checks that a point is canonical.
///
/// https://github.com/dalek-cryptography/curve25519-dalek/issues/380
pub(crate) fn check_point(point: &CompressedEdwardsY) -> bool {
let bytes = point.as_bytes();
point
.decompress()
// Ban points which are either unreduced or -0
.filter(|point| point.compress().as_bytes() == bytes)
.is_some()
}