cuprate/consensus/src/context/hardforks.rs

183 lines
5.3 KiB
Rust
Raw Normal View History

use std::ops::Range;
2023-09-03 22:50:38 +00:00
use tower::ServiceExt;
use tracing::instrument;
use cuprate_consensus_rules::{HFVotes, HFsInfo, HardFork};
use cuprate_types::blockchain::{BCReadRequest, BCResponse};
use crate::{Database, ExtendedConsensusError};
2023-09-03 22:50:38 +00:00
/// The default amount of hard-fork votes to track to decide on activation of a hard-fork.
///
/// ref: <https://cuprate.github.io/monero-docs/consensus_rules/hardforks.html#accepting-a-fork>
2023-09-03 22:50:38 +00:00
const DEFAULT_WINDOW_SIZE: u64 = 10080; // supermajority window check length - a week
/// Configuration for hard-forks.
///
#[derive(Debug, Clone)]
pub struct HardForkConfig {
/// The network we are on.
2024-01-19 00:34:30 +00:00
pub(crate) info: HFsInfo,
/// The amount of votes we are taking into account to decide on a fork activation.
2024-01-19 00:34:30 +00:00
pub(crate) window: u64,
}
impl HardForkConfig {
/// Config for main-net.
pub const fn main_net() -> HardForkConfig {
Self {
info: HFsInfo::main_net(),
window: DEFAULT_WINDOW_SIZE,
}
}
/// Config for stage-net.
pub const fn stage_net() -> HardForkConfig {
Self {
info: HFsInfo::stage_net(),
window: DEFAULT_WINDOW_SIZE,
}
}
/// Config for test-net.
pub const fn test_net() -> HardForkConfig {
Self {
info: HFsInfo::test_net(),
window: DEFAULT_WINDOW_SIZE,
}
}
}
2023-09-03 22:50:38 +00:00
/// A struct that keeps track of the current hard-fork and current votes.
#[derive(Debug, Clone)]
pub struct HardForkState {
/// The current active hard-fork.
2024-01-19 00:34:30 +00:00
pub(crate) current_hardfork: HardFork,
2023-09-03 22:50:38 +00:00
/// The hard-fork config.
2024-01-19 00:34:30 +00:00
pub(crate) config: HardForkConfig,
/// The votes in the current window.
2024-01-19 00:34:30 +00:00
pub(crate) votes: HFVotes,
2023-09-03 22:50:38 +00:00
/// The last block height accounted for.
2024-01-19 00:34:30 +00:00
pub(crate) last_height: u64,
2023-09-03 22:50:38 +00:00
}
impl HardForkState {
/// Initialize the [`HardForkState`] from the specified chain height.
#[instrument(name = "init_hardfork_state", skip(config, database), level = "info")]
pub async fn init_from_chain_height<D: Database + Clone>(
chain_height: u64,
config: HardForkConfig,
mut database: D,
) -> Result<Self, ExtendedConsensusError> {
tracing::info!("Initializing hard-fork state this may take a while.");
2023-09-06 14:54:49 +00:00
let block_start = chain_height.saturating_sub(config.window);
2023-09-03 22:50:38 +00:00
let votes = get_votes_in_range(
database.clone(),
block_start..chain_height,
usize::try_from(config.window).unwrap(),
)
.await?;
2023-09-03 22:50:38 +00:00
if chain_height > config.window {
debug_assert_eq!(votes.total_votes(), config.window)
2023-09-03 22:50:38 +00:00
}
let BCResponse::BlockExtendedHeader(ext_header) = database
.ready()
.await?
.call(BCReadRequest::BlockExtendedHeader(chain_height - 1))
.await?
else {
panic!("Database sent incorrect response!");
};
2023-09-03 22:50:38 +00:00
let current_hardfork =
HardFork::from_version(ext_header.version).expect("Stored block has invalid hardfork");
2023-09-03 22:50:38 +00:00
let mut hfs = HardForkState {
2023-09-03 22:50:38 +00:00
config,
current_hardfork,
votes,
last_height: chain_height - 1,
};
hfs.check_set_new_hf();
tracing::info!(
"Initialized Hfs, current fork: {:?}, {}",
hfs.current_hardfork,
hfs.votes
);
2023-09-03 22:50:38 +00:00
Ok(hfs)
}
/// Add a new block to the cache.
pub fn new_block(&mut self, vote: HardFork, height: u64) {
// We don't _need_ to take in `height` but it's for safety, so we don't silently loose track
// of blocks.
assert_eq!(self.last_height + 1, height);
2023-09-03 22:50:38 +00:00
self.last_height += 1;
tracing::debug!(
"Accounting for new blocks vote, height: {}, vote: {:?}",
self.last_height,
vote
);
// This function remove votes outside the window as well.
2023-09-03 22:50:38 +00:00
self.votes.add_vote_for_hf(&vote);
if height > self.config.window {
debug_assert_eq!(self.votes.total_votes(), self.config.window);
2023-09-03 22:50:38 +00:00
}
self.check_set_new_hf();
2023-09-03 22:50:38 +00:00
}
/// Checks if the next hard-fork should be activated and activates it if it should.
///
/// https://cuprate.github.io/monero-docs/consensus_rules/hardforks.html#accepting-a-fork
fn check_set_new_hf(&mut self) {
2023-12-17 03:13:30 +00:00
self.current_hardfork = self.votes.current_fork(
&self.current_hardfork,
self.last_height + 1,
self.config.window,
&self.config.info,
2023-12-17 03:13:30 +00:00
);
2023-09-03 22:50:38 +00:00
}
/// Returns the current hard-fork.
pub fn current_hardfork(&self) -> HardFork {
self.current_hardfork
}
2023-09-03 22:50:38 +00:00
}
/// Returns the block votes for blocks in the specified range.
#[instrument(name = "get_votes", skip(database))]
async fn get_votes_in_range<D: Database>(
database: D,
2023-09-03 22:50:38 +00:00
block_heights: Range<u64>,
window_size: usize,
) -> Result<HFVotes, ExtendedConsensusError> {
let mut votes = HFVotes::new(window_size);
2023-09-03 22:50:38 +00:00
let BCResponse::BlockExtendedHeaderInRange(vote_list) = database
.oneshot(BCReadRequest::BlockExtendedHeaderInRange(block_heights))
.await?
else {
panic!("Database sent incorrect response!");
};
2023-09-03 22:50:38 +00:00
for hf_info in vote_list.into_iter() {
votes.add_vote_for_hf(&HardFork::from_vote(hf_info.vote));
2023-09-03 22:50:38 +00:00
}
Ok(votes)
}