mirror of
https://github.com/Cuprate/cuprate.git
synced 2025-01-08 20:09:44 +00:00
fmt + clippy + fix tests
This commit is contained in:
parent
b0588fad2b
commit
376a41deb2
10 changed files with 101 additions and 106 deletions
|
@ -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"
|
|
@ -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"]}
|
|
@ -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
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
69
consensus/rules/src/hard_forks/tests.rs
Normal file
69
consensus/rules/src/hard_forks/tests.rs
Normal 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));
|
||||||
|
}
|
||||||
|
}
|
|
@ -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;
|
||||||
|
|
|
@ -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()
|
||||||
|
|
|
@ -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::{
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -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()
|
|
||||||
}
|
|
||||||
|
|
Loading…
Reference in a new issue