2024-05-31 00:52:12 +00:00
|
|
|
//! Context Task
|
|
|
|
//!
|
|
|
|
//! This module contains the async task that handles keeping track of blockchain context.
|
|
|
|
//! It holds all the context caches and handles [`tower::Service`] requests.
|
|
|
|
//!
|
|
|
|
use futures::channel::oneshot;
|
|
|
|
use tokio::sync::mpsc;
|
|
|
|
use tower::ServiceExt;
|
|
|
|
use tracing::Instrument;
|
|
|
|
|
|
|
|
use cuprate_consensus_rules::blocks::ContextToVerifyBlock;
|
2024-09-02 17:09:52 +00:00
|
|
|
use cuprate_helper::cast::u64_to_usize;
|
2024-07-29 00:13:08 +00:00
|
|
|
use cuprate_types::{
|
2024-08-05 20:47:30 +00:00
|
|
|
blockchain::{BlockchainReadRequest, BlockchainResponse},
|
2024-07-29 00:13:08 +00:00
|
|
|
Chain,
|
|
|
|
};
|
2024-05-31 00:52:12 +00:00
|
|
|
|
2024-07-29 00:13:08 +00:00
|
|
|
use crate::{
|
|
|
|
context::{
|
|
|
|
alt_chains::{get_alt_chain_difficulty_cache, get_alt_chain_weight_cache, AltChainMap},
|
|
|
|
difficulty, hardforks, rx_vms, weight, BlockChainContext, BlockChainContextRequest,
|
|
|
|
BlockChainContextResponse, ContextConfig, RawBlockChainContext, ValidityToken,
|
|
|
|
BLOCKCHAIN_TIMESTAMP_CHECK_WINDOW,
|
|
|
|
},
|
|
|
|
Database, ExtendedConsensusError,
|
2024-05-31 00:52:12 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// A request from the context service to the context task.
|
|
|
|
pub(super) struct ContextTaskRequest {
|
|
|
|
/// The request.
|
|
|
|
pub req: BlockChainContextRequest,
|
|
|
|
/// The response channel.
|
|
|
|
pub tx: oneshot::Sender<Result<BlockChainContextResponse, tower::BoxError>>,
|
|
|
|
/// The tracing span of the requester.
|
|
|
|
pub span: tracing::Span,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The Context task that keeps the blockchain context and handles requests.
|
2024-07-29 00:13:08 +00:00
|
|
|
pub struct ContextTask<D: Database> {
|
2024-05-31 00:52:12 +00:00
|
|
|
/// A token used to invalidate previous contexts when a new
|
|
|
|
/// block is added to the chain.
|
|
|
|
current_validity_token: ValidityToken,
|
|
|
|
|
|
|
|
/// The difficulty cache.
|
|
|
|
difficulty_cache: difficulty::DifficultyCache,
|
|
|
|
/// The weight cache.
|
|
|
|
weight_cache: weight::BlockWeightsCache,
|
|
|
|
/// The RX VM cache.
|
2024-09-02 17:10:45 +00:00
|
|
|
rx_vm_cache: rx_vms::RandomXVmCache,
|
2024-05-31 00:52:12 +00:00
|
|
|
/// The hard-fork state cache.
|
|
|
|
hardfork_state: hardforks::HardForkState,
|
|
|
|
|
2024-07-29 00:13:08 +00:00
|
|
|
alt_chain_cache_map: AltChainMap,
|
|
|
|
|
2024-05-31 00:52:12 +00:00
|
|
|
/// The current chain height.
|
2024-08-06 23:48:53 +00:00
|
|
|
chain_height: usize,
|
2024-05-31 00:52:12 +00:00
|
|
|
/// The top block hash.
|
|
|
|
top_block_hash: [u8; 32],
|
|
|
|
/// The total amount of coins generated.
|
|
|
|
already_generated_coins: u64,
|
2024-07-29 00:13:08 +00:00
|
|
|
|
|
|
|
database: D,
|
2024-05-31 00:52:12 +00:00
|
|
|
}
|
|
|
|
|
2024-07-29 00:13:08 +00:00
|
|
|
impl<D: Database + Clone + Send + 'static> ContextTask<D> {
|
2024-05-31 00:52:12 +00:00
|
|
|
/// Initialize the [`ContextTask`], this will need to pull a lot of data from the database so may take a
|
|
|
|
/// while to complete.
|
2024-07-29 00:13:08 +00:00
|
|
|
pub async fn init_context(
|
2024-05-31 00:52:12 +00:00
|
|
|
cfg: ContextConfig,
|
|
|
|
mut database: D,
|
2024-07-29 00:13:08 +00:00
|
|
|
) -> Result<Self, ExtendedConsensusError> {
|
2024-05-31 00:52:12 +00:00
|
|
|
let ContextConfig {
|
|
|
|
difficulty_cfg,
|
|
|
|
weights_config,
|
|
|
|
hard_fork_cfg,
|
|
|
|
} = cfg;
|
|
|
|
|
|
|
|
tracing::debug!("Initialising blockchain context");
|
|
|
|
|
2024-08-05 20:47:30 +00:00
|
|
|
let BlockchainResponse::ChainHeight(chain_height, top_block_hash) = database
|
2024-05-31 00:52:12 +00:00
|
|
|
.ready()
|
|
|
|
.await?
|
2024-08-05 20:47:30 +00:00
|
|
|
.call(BlockchainReadRequest::ChainHeight)
|
2024-05-31 00:52:12 +00:00
|
|
|
.await?
|
|
|
|
else {
|
|
|
|
panic!("Database sent incorrect response!");
|
|
|
|
};
|
|
|
|
|
2024-08-05 20:47:30 +00:00
|
|
|
let BlockchainResponse::GeneratedCoins(already_generated_coins) = database
|
2024-05-31 00:52:12 +00:00
|
|
|
.ready()
|
|
|
|
.await?
|
2024-08-05 20:47:30 +00:00
|
|
|
.call(BlockchainReadRequest::GeneratedCoins(chain_height - 1))
|
2024-05-31 00:52:12 +00:00
|
|
|
.await?
|
|
|
|
else {
|
|
|
|
panic!("Database sent incorrect response!");
|
|
|
|
};
|
|
|
|
|
|
|
|
let db = database.clone();
|
|
|
|
let hardfork_state_handle = tokio::spawn(async move {
|
|
|
|
hardforks::HardForkState::init_from_chain_height(chain_height, hard_fork_cfg, db).await
|
|
|
|
});
|
|
|
|
|
|
|
|
let db = database.clone();
|
|
|
|
let difficulty_cache_handle = tokio::spawn(async move {
|
2024-07-29 00:13:08 +00:00
|
|
|
difficulty::DifficultyCache::init_from_chain_height(
|
|
|
|
chain_height,
|
|
|
|
difficulty_cfg,
|
|
|
|
db,
|
|
|
|
Chain::Main,
|
|
|
|
)
|
|
|
|
.await
|
2024-05-31 00:52:12 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
let db = database.clone();
|
|
|
|
let weight_cache_handle = tokio::spawn(async move {
|
2024-07-29 00:13:08 +00:00
|
|
|
weight::BlockWeightsCache::init_from_chain_height(
|
|
|
|
chain_height,
|
|
|
|
weights_config,
|
|
|
|
db,
|
|
|
|
Chain::Main,
|
|
|
|
)
|
|
|
|
.await
|
2024-05-31 00:52:12 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// Wait for the hardfork state to finish first as we need it to start the randomX VM cache.
|
|
|
|
let hardfork_state = hardfork_state_handle.await.unwrap()?;
|
|
|
|
let current_hf = hardfork_state.current_hardfork();
|
|
|
|
|
|
|
|
let db = database.clone();
|
|
|
|
let rx_seed_handle = tokio::spawn(async move {
|
2024-09-02 17:10:45 +00:00
|
|
|
rx_vms::RandomXVmCache::init_from_chain_height(chain_height, ¤t_hf, db).await
|
2024-05-31 00:52:12 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
let context_svc = ContextTask {
|
|
|
|
current_validity_token: ValidityToken::new(),
|
|
|
|
difficulty_cache: difficulty_cache_handle.await.unwrap()?,
|
|
|
|
weight_cache: weight_cache_handle.await.unwrap()?,
|
|
|
|
rx_vm_cache: rx_seed_handle.await.unwrap()?,
|
|
|
|
hardfork_state,
|
2024-07-29 00:13:08 +00:00
|
|
|
alt_chain_cache_map: AltChainMap::new(),
|
2024-05-31 00:52:12 +00:00
|
|
|
chain_height,
|
|
|
|
already_generated_coins,
|
|
|
|
top_block_hash,
|
2024-07-29 00:13:08 +00:00
|
|
|
database,
|
2024-05-31 00:52:12 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Ok(context_svc)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Handles a [`BlockChainContextRequest`] and returns a [`BlockChainContextResponse`].
|
|
|
|
pub async fn handle_req(
|
|
|
|
&mut self,
|
|
|
|
req: BlockChainContextRequest,
|
|
|
|
) -> Result<BlockChainContextResponse, tower::BoxError> {
|
|
|
|
Ok(match req {
|
|
|
|
BlockChainContextRequest::GetContext => {
|
|
|
|
tracing::debug!("Getting blockchain context");
|
|
|
|
|
|
|
|
let current_hf = self.hardfork_state.current_hardfork();
|
|
|
|
|
|
|
|
BlockChainContextResponse::Context(BlockChainContext {
|
|
|
|
validity_token: self.current_validity_token.clone(),
|
|
|
|
raw: RawBlockChainContext {
|
|
|
|
context_to_verify_block: ContextToVerifyBlock {
|
|
|
|
median_weight_for_block_reward: self
|
|
|
|
.weight_cache
|
|
|
|
.median_for_block_reward(¤t_hf),
|
|
|
|
effective_median_weight: self
|
|
|
|
.weight_cache
|
|
|
|
.effective_median_block_weight(¤t_hf),
|
|
|
|
top_hash: self.top_block_hash,
|
2024-09-02 17:09:52 +00:00
|
|
|
median_block_timestamp: self
|
|
|
|
.difficulty_cache
|
|
|
|
.median_timestamp(u64_to_usize(BLOCKCHAIN_TIMESTAMP_CHECK_WINDOW)),
|
2024-05-31 00:52:12 +00:00
|
|
|
chain_height: self.chain_height,
|
|
|
|
current_hf,
|
|
|
|
next_difficulty: self.difficulty_cache.next_difficulty(¤t_hf),
|
|
|
|
already_generated_coins: self.already_generated_coins,
|
|
|
|
},
|
|
|
|
cumulative_difficulty: self.difficulty_cache.cumulative_difficulty(),
|
|
|
|
median_long_term_weight: self.weight_cache.median_long_term_weight(),
|
|
|
|
top_block_timestamp: self.difficulty_cache.top_block_timestamp(),
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
2024-06-25 00:55:04 +00:00
|
|
|
BlockChainContextRequest::GetCurrentRxVm => {
|
|
|
|
BlockChainContextResponse::RxVms(self.rx_vm_cache.get_vms().await)
|
|
|
|
}
|
2024-05-31 00:52:12 +00:00
|
|
|
BlockChainContextRequest::BatchGetDifficulties(blocks) => {
|
|
|
|
tracing::debug!("Getting batch difficulties len: {}", blocks.len() + 1);
|
|
|
|
|
|
|
|
let next_diffs = self
|
|
|
|
.difficulty_cache
|
|
|
|
.next_difficulties(blocks, &self.hardfork_state.current_hardfork());
|
|
|
|
BlockChainContextResponse::BatchDifficulties(next_diffs)
|
|
|
|
}
|
|
|
|
BlockChainContextRequest::NewRXVM(vm) => {
|
|
|
|
tracing::debug!("Adding randomX VM to cache.");
|
|
|
|
|
|
|
|
self.rx_vm_cache.add_vm(vm);
|
|
|
|
BlockChainContextResponse::Ok
|
|
|
|
}
|
|
|
|
BlockChainContextRequest::Update(new) => {
|
|
|
|
tracing::debug!(
|
|
|
|
"Updating blockchain cache with new block, height: {}",
|
|
|
|
new.height
|
|
|
|
);
|
|
|
|
// Cancel the validity token and replace it with a new one.
|
|
|
|
std::mem::replace(&mut self.current_validity_token, ValidityToken::new())
|
|
|
|
.set_data_invalid();
|
|
|
|
|
|
|
|
self.difficulty_cache.new_block(
|
|
|
|
new.height,
|
|
|
|
new.timestamp,
|
|
|
|
new.cumulative_difficulty,
|
|
|
|
);
|
|
|
|
|
|
|
|
self.weight_cache
|
|
|
|
.new_block(new.height, new.weight, new.long_term_weight);
|
|
|
|
|
|
|
|
self.hardfork_state.new_block(new.vote, new.height);
|
|
|
|
|
2024-06-25 00:55:04 +00:00
|
|
|
self.rx_vm_cache.new_block(new.height, &new.block_hash);
|
2024-05-31 00:52:12 +00:00
|
|
|
|
|
|
|
self.chain_height = new.height + 1;
|
|
|
|
self.top_block_hash = new.block_hash;
|
|
|
|
self.already_generated_coins = self
|
|
|
|
.already_generated_coins
|
|
|
|
.saturating_add(new.generated_coins);
|
|
|
|
|
|
|
|
BlockChainContextResponse::Ok
|
|
|
|
}
|
2024-07-29 00:13:08 +00:00
|
|
|
BlockChainContextRequest::PopBlocks { numb_blocks } => {
|
|
|
|
assert!(numb_blocks < self.chain_height);
|
|
|
|
|
|
|
|
self.difficulty_cache
|
|
|
|
.pop_blocks_main_chain(numb_blocks, self.database.clone())
|
|
|
|
.await?;
|
|
|
|
self.weight_cache
|
|
|
|
.pop_blocks_main_chain(numb_blocks, self.database.clone())
|
|
|
|
.await?;
|
|
|
|
self.rx_vm_cache
|
|
|
|
.pop_blocks_main_chain(self.chain_height - numb_blocks - 1);
|
|
|
|
self.hardfork_state
|
|
|
|
.pop_blocks_main_chain(numb_blocks, self.database.clone())
|
|
|
|
.await?;
|
|
|
|
|
|
|
|
self.alt_chain_cache_map.clear();
|
|
|
|
|
|
|
|
self.chain_height -= numb_blocks;
|
|
|
|
|
2024-08-05 20:47:30 +00:00
|
|
|
let BlockchainResponse::GeneratedCoins(already_generated_coins) = self
|
2024-07-29 00:13:08 +00:00
|
|
|
.database
|
|
|
|
.ready()
|
|
|
|
.await?
|
2024-08-05 20:47:30 +00:00
|
|
|
.call(BlockchainReadRequest::GeneratedCoins(self.chain_height - 1))
|
2024-07-29 00:13:08 +00:00
|
|
|
.await?
|
|
|
|
else {
|
|
|
|
panic!("Database sent incorrect response!");
|
|
|
|
};
|
|
|
|
|
2024-08-05 20:47:30 +00:00
|
|
|
let BlockchainResponse::BlockHash(top_block_hash) = self
|
2024-07-29 00:13:08 +00:00
|
|
|
.database
|
|
|
|
.ready()
|
|
|
|
.await?
|
2024-08-05 20:47:30 +00:00
|
|
|
.call(BlockchainReadRequest::BlockHash(
|
|
|
|
self.chain_height - 1,
|
|
|
|
Chain::Main,
|
|
|
|
))
|
2024-07-29 00:13:08 +00:00
|
|
|
.await?
|
|
|
|
else {
|
|
|
|
panic!("Database returned incorrect response!");
|
|
|
|
};
|
|
|
|
|
|
|
|
self.already_generated_coins = already_generated_coins;
|
|
|
|
self.top_block_hash = top_block_hash;
|
|
|
|
|
|
|
|
std::mem::replace(&mut self.current_validity_token, ValidityToken::new())
|
|
|
|
.set_data_invalid();
|
|
|
|
|
|
|
|
BlockChainContextResponse::Ok
|
|
|
|
}
|
|
|
|
BlockChainContextRequest::ClearAltCache => {
|
|
|
|
self.alt_chain_cache_map.clear();
|
|
|
|
|
|
|
|
BlockChainContextResponse::Ok
|
|
|
|
}
|
|
|
|
BlockChainContextRequest::AltChainContextCache { prev_id, _token } => {
|
|
|
|
BlockChainContextResponse::AltChainContextCache(
|
|
|
|
self.alt_chain_cache_map
|
|
|
|
.get_alt_chain_context(prev_id, &mut self.database)
|
|
|
|
.await?,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
BlockChainContextRequest::AltChainDifficultyCache { prev_id, _token } => {
|
|
|
|
BlockChainContextResponse::AltChainDifficultyCache(
|
|
|
|
get_alt_chain_difficulty_cache(
|
|
|
|
prev_id,
|
|
|
|
&self.difficulty_cache,
|
|
|
|
self.database.clone(),
|
|
|
|
)
|
|
|
|
.await?,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
BlockChainContextRequest::AltChainWeightCache { prev_id, _token } => {
|
|
|
|
BlockChainContextResponse::AltChainWeightCache(
|
|
|
|
get_alt_chain_weight_cache(prev_id, &self.weight_cache, self.database.clone())
|
|
|
|
.await?,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
BlockChainContextRequest::AltChainRxVM {
|
|
|
|
height,
|
|
|
|
chain,
|
|
|
|
_token,
|
|
|
|
} => BlockChainContextResponse::AltChainRxVM(
|
|
|
|
self.rx_vm_cache
|
|
|
|
.get_alt_vm(height, chain, &mut self.database)
|
|
|
|
.await?,
|
|
|
|
),
|
|
|
|
BlockChainContextRequest::AddAltChainContextCache {
|
|
|
|
prev_id,
|
|
|
|
cache,
|
|
|
|
_token,
|
|
|
|
} => {
|
|
|
|
self.alt_chain_cache_map.add_alt_cache(prev_id, cache);
|
|
|
|
BlockChainContextResponse::Ok
|
|
|
|
}
|
2024-05-31 00:52:12 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Run the [`ContextTask`], the task will listen for requests on the passed in channel. When the channel closes the
|
|
|
|
/// task will finish.
|
|
|
|
pub async fn run(mut self, mut rx: mpsc::Receiver<ContextTaskRequest>) {
|
|
|
|
while let Some(req) = rx.recv().await {
|
|
|
|
let res = self.handle_req(req.req).instrument(req.span).await;
|
|
|
|
let _ = req.tx.send(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
tracing::info!("Shutting down blockchain context task.");
|
|
|
|
}
|
|
|
|
}
|