From 37b1b0bff7da31bd5b883a094829c4395ad0333b Mon Sep 17 00:00:00 2001 From: Anton Puhach Date: Thu, 4 Jan 2024 12:53:31 +0100 Subject: [PATCH] refactor: move ChainUpdate to a separate file (#10380) This PR moves `ChainUpdate` to its own module as it is without any additional refactoring and making all currently used fields and method `pub(crate)`. I'm planning to fix that in a separate PR to make it easier to review. Part of https://github.com/near/nearcore/issues/10275. --- chain/chain/src/chain.rs | 1038 +----------------------------- chain/chain/src/chain_update.rs | 1057 +++++++++++++++++++++++++++++++ chain/chain/src/lib.rs | 4 +- 3 files changed, 1068 insertions(+), 1031 deletions(-) create mode 100644 chain/chain/src/chain_update.rs diff --git a/chain/chain/src/chain.rs b/chain/chain/src/chain.rs index 0c64f555375..0978e77aa58 100644 --- a/chain/chain/src/chain.rs +++ b/chain/chain/src/chain.rs @@ -2,9 +2,9 @@ use crate::block_processing_utils::{ BlockPreprocessInfo, BlockProcessingArtifact, BlocksInProcessing, DoneApplyChunkCallback, }; use crate::blocks_delay_tracker::BlocksDelayTracker; +use crate::chain_update::ChainUpdate; use crate::crypto_hash_timer::CryptoHashTimer; use crate::lightclient::get_epoch_block_producers_view; -use crate::metrics::{SHARD_LAYOUT_NUM_SHARDS, SHARD_LAYOUT_VERSION}; use crate::migrations::check_if_block_is_first_with_chunk_of_version; use crate::missing_chunks::MissingChunksPool; use crate::orphan::{Orphan, OrphanBlockPool}; @@ -13,14 +13,13 @@ use crate::state_snapshot_actor::SnapshotCallbacks; use crate::store::{ChainStore, ChainStoreAccess, ChainStoreUpdate}; use crate::types::{ - AcceptedBlock, ApplySplitStateResultOrStateChanges, ApplyTransactionResult, - ApplyTransactionsBlockContext, ApplyTransactionsChunkContext, BlockEconomicsConfig, - ChainConfig, RuntimeAdapter, RuntimeStorageConfig, StorageDataSource, + AcceptedBlock, ApplyTransactionResult, ApplyTransactionsBlockContext, BlockEconomicsConfig, + ChainConfig, RuntimeAdapter, StorageDataSource, }; use crate::update_shard::{ apply_new_chunk, process_missing_chunks_range, process_shard_update, NewChunkData, - NewChunkResult, OldChunkData, OldChunkResult, ShardBlockUpdateResult, ShardContext, - ShardUpdateReason, ShardUpdateResult, StateSplitData, StateSplitResult, StorageContext, + NewChunkResult, OldChunkData, ShardContext, ShardUpdateReason, ShardUpdateResult, + StateSplitData, StorageContext, }; use crate::validate::{ validate_challenge, validate_chunk_proofs, validate_chunk_with_chunk_extra, @@ -52,7 +51,7 @@ use near_primitives::challenge::{ }; use near_primitives::checked_feature; #[cfg(feature = "new_epoch_sync")] -use near_primitives::epoch_manager::{block_info::BlockInfo, epoch_sync::EpochSyncInfo}; +use near_primitives::epoch_manager::epoch_sync::EpochSyncInfo; #[cfg(feature = "new_epoch_sync")] use near_primitives::errors::epoch_sync::EpochSyncHashType; use near_primitives::errors::EpochError; @@ -62,9 +61,7 @@ use near_primitives::merkle::{ }; use near_primitives::receipt::Receipt; use near_primitives::sandbox::state_patch::SandboxStatePatch; -use near_primitives::shard_layout::{ - account_id_to_shard_id, account_id_to_shard_uid, ShardLayout, ShardUId, -}; +use near_primitives::shard_layout::{account_id_to_shard_id, ShardLayout, ShardUId}; use near_primitives::sharding::{ ChunkHash, ChunkHashHeight, EncodedShardChunk, ReceiptList, ReceiptProof, ShardChunk, ShardChunkHeader, ShardInfo, ShardProof, StateSyncInfo, @@ -79,7 +76,7 @@ use near_primitives::transaction::{ExecutionOutcomeWithIdAndProof, SignedTransac use near_primitives::types::chunk_extra::ChunkExtra; use near_primitives::types::{ AccountId, Balance, BlockExtra, BlockHeight, BlockHeightDelta, EpochId, MerkleHash, NumBlocks, - NumShards, ShardId, StateRoot, + ShardId, StateRoot, }; use near_primitives::unwrap_or_return; #[cfg(feature = "new_epoch_sync")] @@ -4564,1025 +4561,6 @@ impl Chain { } } -/// Chain update helper, contains information that is needed to process block -/// and decide to accept it or reject it. -/// If rejected nothing will be updated in underlying storage. -/// Safe to stop process mid way (Ctrl+C or crash). -pub struct ChainUpdate<'a> { - epoch_manager: Arc, - shard_tracker: ShardTracker, - runtime_adapter: Arc, - chain_store_update: ChainStoreUpdate<'a>, - doomslug_threshold_mode: DoomslugThresholdMode, - #[allow(unused)] - transaction_validity_period: BlockHeightDelta, -} - -impl<'a> ChainUpdate<'a> { - pub fn new( - store: &'a mut ChainStore, - epoch_manager: Arc, - shard_tracker: ShardTracker, - runtime_adapter: Arc, - doomslug_threshold_mode: DoomslugThresholdMode, - transaction_validity_period: BlockHeightDelta, - ) -> Self { - let chain_store_update: ChainStoreUpdate<'_> = store.store_update(); - Self::new_impl( - epoch_manager, - shard_tracker, - runtime_adapter, - doomslug_threshold_mode, - transaction_validity_period, - chain_store_update, - ) - } - - fn new_impl( - epoch_manager: Arc, - shard_tracker: ShardTracker, - runtime_adapter: Arc, - doomslug_threshold_mode: DoomslugThresholdMode, - transaction_validity_period: BlockHeightDelta, - chain_store_update: ChainStoreUpdate<'a>, - ) -> Self { - ChainUpdate { - epoch_manager, - shard_tracker, - runtime_adapter, - chain_store_update, - doomslug_threshold_mode, - transaction_validity_period, - } - } - - /// Commit changes to the chain into the database. - pub fn commit(self) -> Result<(), Error> { - self.chain_store_update.commit() - } - - /// For all the outgoing receipts generated in block `hash` at the shards we - /// are tracking in this epoch, save a mapping from receipt ids to the - /// destination shard ids that the receipt will be sent to in the next - /// block. - /// - /// Note that this function should be called after `save_block` is called on - /// this block because it requires that the block info is available in - /// EpochManager, otherwise it will return an error. - fn save_receipt_id_to_shard_id_for_block( - &mut self, - account_id: Option<&AccountId>, - hash: &CryptoHash, - prev_hash: &CryptoHash, - shard_ids: &[ShardId], - ) -> Result<(), Error> { - let mut list = vec![]; - for &shard_id in shard_ids { - if self.shard_tracker.care_about_shard(account_id, prev_hash, shard_id, true) { - list.push(self.get_receipt_id_to_shard_id(hash, shard_id)?); - } - } - for map in list { - for (receipt_id, shard_id) in map { - self.chain_store_update.save_receipt_id_to_shard_id(receipt_id, shard_id); - } - } - Ok(()) - } - - /// Returns a mapping from the receipt id to the destination shard id. - fn get_receipt_id_to_shard_id( - &mut self, - hash: &CryptoHash, - shard_id: u64, - ) -> Result, Error> { - let outgoing_receipts = self.chain_store_update.get_outgoing_receipts(hash, shard_id); - let outgoing_receipts = if let Ok(outgoing_receipts) = outgoing_receipts { - outgoing_receipts - } else { - return Ok(HashMap::new()); - }; - let shard_layout = self.epoch_manager.get_shard_layout_from_prev_block(hash)?; - let outgoing_receipts = outgoing_receipts - .iter() - .map(|receipt| { - (receipt.receipt_id, account_id_to_shard_id(&receipt.receiver_id, &shard_layout)) - }) - .collect(); - Ok(outgoing_receipts) - } - - fn apply_chunk_postprocessing( - &mut self, - block: &Block, - apply_results: Vec, - ) -> Result<(), Error> { - let _span = tracing::debug_span!(target: "chain", "apply_chunk_postprocessing").entered(); - for result in apply_results { - match result { - ShardUpdateResult::Stateful(result) => { - self.process_apply_chunk_result(block, result)? - } - ShardUpdateResult::Stateless(results) => { - for (block_hash, shard_uid, chunk_extra) in results { - let expected_chunk_extra = - self.chain_store_update.get_chunk_extra(&block_hash, &shard_uid)?; - assert_eq!( - &chunk_extra, - expected_chunk_extra.as_ref(), - "For stateless validation, chunk extras for block {} and shard {} do not match", - block_hash, - shard_uid - ); - } - } - } - } - Ok(()) - } - - /// Postprocess split state results or state changes, do the necessary update on chain - /// for split state results: store the chunk extras and trie changes for the split states - /// for state changes, store the state changes for splitting states - fn process_split_state( - &mut self, - block: &Block, - shard_uid: &ShardUId, - apply_results_or_state_changes: ApplySplitStateResultOrStateChanges, - ) -> Result<(), Error> { - let block_hash = block.hash(); - let prev_hash = block.header().prev_hash(); - let height = block.header().height(); - match apply_results_or_state_changes { - ApplySplitStateResultOrStateChanges::ApplySplitStateResults(mut results) => { - tracing::debug!(target: "resharding", height, ?shard_uid, "process_split_state apply"); - - // Sort the results so that the gas reassignment is deterministic. - results.sort_unstable_by_key(|r| r.shard_uid); - // Drop the mutability as we no longer need it. - let results = results; - - // Split validator_proposals, gas_burnt, balance_burnt to each split shard - // and store the chunk extra for split shards - // Note that here we do not split outcomes by the new shard layout, we simply store - // the outcome_root from the parent shard. This is because outcome proofs are - // generated per shard using the old shard layout and stored in the database. - // For these proofs to work, we must store the outcome root per shard - // using the old shard layout instead of the new shard layout - let chunk_extra = self.chain_store_update.get_chunk_extra(block_hash, shard_uid)?; - let next_epoch_shard_layout = { - let epoch_id = - self.epoch_manager.get_next_epoch_id_from_prev_block(prev_hash)?; - self.epoch_manager.get_shard_layout(&epoch_id)? - }; - - let mut validator_proposals_by_shard: HashMap<_, Vec<_>> = HashMap::new(); - for validator_proposal in chunk_extra.validator_proposals() { - let shard_uid = account_id_to_shard_uid( - validator_proposal.account_id(), - &next_epoch_shard_layout, - ); - validator_proposals_by_shard - .entry(shard_uid) - .or_default() - .push(validator_proposal); - } - - let num_split_shards = next_epoch_shard_layout - .get_split_shard_uids(shard_uid.shard_id()) - .unwrap_or_else(|| panic!("invalid shard layout {:?}", next_epoch_shard_layout)) - .len() as NumShards; - - let total_gas_used = chunk_extra.gas_used(); - let total_balance_burnt = chunk_extra.balance_burnt(); - - // The gas remainder, the split shards will be reassigned one - // unit each until its depleted. - let mut gas_res = total_gas_used % num_split_shards; - // The gas quotient, the split shards will be reassigned the - // full value each. - let gas_split = total_gas_used / num_split_shards; - - // The balance remainder, the split shards will be reassigned one - // unit each until its depleted. - let mut balance_res = (total_balance_burnt % num_split_shards as u128) as NumShards; - // The balance quotient, the split shards will be reassigned the - // full value each. - let balance_split = total_balance_burnt / (num_split_shards as u128); - - let gas_limit = chunk_extra.gas_limit(); - let outcome_root = *chunk_extra.outcome_root(); - - let mut sum_gas_used = 0; - let mut sum_balance_burnt = 0; - - // The gas and balance distribution assumes that we have a result for every split shard. - // TODO(resharding) make sure that is the case. - assert_eq!(num_split_shards, results.len() as u64); - - for result in results { - let gas_burnt = if gas_res > 0 { - gas_res -= 1; - gas_split + 1 - } else { - gas_split - }; - - let balance_burnt = if balance_res > 0 { - balance_res -= 1; - balance_split + 1 - } else { - balance_split - }; - - let new_chunk_extra = ChunkExtra::new( - &result.new_root, - outcome_root, - validator_proposals_by_shard.remove(&result.shard_uid).unwrap_or_default(), - gas_burnt, - gas_limit, - balance_burnt, - ); - sum_gas_used += gas_burnt; - sum_balance_burnt += balance_burnt; - - // TODO(#9430): Support manager.save_flat_state_changes and manager.update_flat_storage_for_shard - // functions to be a part of the same chain_store_update - let flat_storage_manager = self.runtime_adapter.get_flat_storage_manager(); - let store_update = flat_storage_manager.save_flat_state_changes( - *block_hash, - *prev_hash, - block.header().height(), - result.shard_uid, - result.trie_changes.state_changes(), - )?; - flat_storage_manager.update_flat_storage_for_shard(*shard_uid, block)?; - self.chain_store_update.merge(store_update); - - self.chain_store_update.save_chunk_extra( - block_hash, - &result.shard_uid, - new_chunk_extra, - ); - self.chain_store_update.save_trie_changes(result.trie_changes); - } - assert_eq!(sum_gas_used, total_gas_used); - assert_eq!(sum_balance_burnt, total_balance_burnt); - } - ApplySplitStateResultOrStateChanges::StateChangesForSplitStates(state_changes) => { - tracing::debug!(target: "resharding", height, ?shard_uid, "process_split_state store"); - self.chain_store_update.add_state_changes_for_split_states( - *block_hash, - shard_uid.shard_id(), - state_changes, - ); - } - } - Ok(()) - } - - /// Processed results of applying chunk - fn process_apply_chunk_result( - &mut self, - block: &Block, - result: ShardBlockUpdateResult, - ) -> Result<(), Error> { - let block_hash = block.hash(); - let prev_hash = block.header().prev_hash(); - let height = block.header().height(); - match result { - ShardBlockUpdateResult::NewChunk(NewChunkResult { - gas_limit, - shard_uid, - apply_result, - apply_split_result_or_state_changes, - }) => { - let (outcome_root, outcome_paths) = - ApplyTransactionResult::compute_outcomes_proof(&apply_result.outcomes); - let shard_id = shard_uid.shard_id(); - - // Save state root after applying transactions. - self.chain_store_update.save_chunk_extra( - block_hash, - &shard_uid, - ChunkExtra::new( - &apply_result.new_root, - outcome_root, - apply_result.validator_proposals, - apply_result.total_gas_burnt, - gas_limit, - apply_result.total_balance_burnt, - ), - ); - - let flat_storage_manager = self.runtime_adapter.get_flat_storage_manager(); - let store_update = flat_storage_manager.save_flat_state_changes( - *block_hash, - *prev_hash, - height, - shard_uid, - apply_result.trie_changes.state_changes(), - )?; - self.chain_store_update.merge(store_update); - - self.chain_store_update.save_trie_changes(apply_result.trie_changes); - self.chain_store_update.save_outgoing_receipt( - block_hash, - shard_id, - apply_result.outgoing_receipts, - ); - // Save receipt and transaction results. - self.chain_store_update.save_outcomes_with_proofs( - block_hash, - shard_id, - apply_result.outcomes, - outcome_paths, - ); - if let Some(apply_results_or_state_changes) = apply_split_result_or_state_changes { - self.process_split_state(block, &shard_uid, apply_results_or_state_changes)?; - } - } - ShardBlockUpdateResult::OldChunk(OldChunkResult { - shard_uid, - apply_result, - apply_split_result_or_state_changes, - }) => { - let old_extra = self.chain_store_update.get_chunk_extra(prev_hash, &shard_uid)?; - - let mut new_extra = ChunkExtra::clone(&old_extra); - *new_extra.state_root_mut() = apply_result.new_root; - - let flat_storage_manager = self.runtime_adapter.get_flat_storage_manager(); - let store_update = flat_storage_manager.save_flat_state_changes( - *block_hash, - *prev_hash, - height, - shard_uid, - apply_result.trie_changes.state_changes(), - )?; - self.chain_store_update.merge(store_update); - - self.chain_store_update.save_chunk_extra(block_hash, &shard_uid, new_extra); - self.chain_store_update.save_trie_changes(apply_result.trie_changes); - - if let Some(apply_results_or_state_changes) = apply_split_result_or_state_changes { - self.process_split_state(block, &shard_uid, apply_results_or_state_changes)?; - } - } - ShardBlockUpdateResult::StateSplit(StateSplitResult { shard_uid, results }) => { - self.chain_store_update - .remove_state_changes_for_split_states(*block.hash(), shard_uid.shard_id()); - self.process_split_state( - block, - &shard_uid, - ApplySplitStateResultOrStateChanges::ApplySplitStateResults(results), - )?; - } - }; - Ok(()) - } - - /// This is the last step of process_block_single, where we take the preprocess block info - /// apply chunk results and store the results on chain. - fn postprocess_block( - &mut self, - me: &Option, - block: &Block, - block_preprocess_info: BlockPreprocessInfo, - apply_chunks_results: Vec<(ShardId, Result)>, - ) -> Result, Error> { - let shard_ids = self.epoch_manager.shard_ids(block.header().epoch_id())?; - let prev_hash = block.header().prev_hash(); - let results = apply_chunks_results.into_iter().map(|(shard_id, x)| { - if let Err(err) = &x { - warn!(target: "chain", shard_id, hash = %block.hash(), %err, "Error in applying chunk for block"); - } - x - }).collect::, Error>>()?; - self.apply_chunk_postprocessing(block, results)?; - - let BlockPreprocessInfo { - is_caught_up, - state_sync_info, - incoming_receipts, - challenges_result, - challenged_blocks, - .. - } = block_preprocess_info; - - if !is_caught_up { - debug!(target: "chain", %prev_hash, hash = %*block.hash(), "Add block to catch up"); - self.chain_store_update.add_block_to_catchup(*prev_hash, *block.hash()); - } - - for (shard_id, receipt_proofs) in incoming_receipts { - self.chain_store_update.save_incoming_receipt( - block.hash(), - shard_id, - Arc::new(receipt_proofs), - ); - } - if let Some(state_sync_info) = state_sync_info { - self.chain_store_update.add_state_sync_info(state_sync_info); - } - - self.chain_store_update.save_block_extra(block.hash(), BlockExtra { challenges_result }); - for block_hash in challenged_blocks { - self.mark_block_as_challenged(&block_hash, Some(block.hash()))?; - } - - self.chain_store_update.save_block_header(block.header().clone())?; - self.update_header_head_if_not_challenged(block.header())?; - - // If block checks out, record validator proposals for given block. - let last_final_block = block.header().last_final_block(); - let last_finalized_height = if last_final_block == &CryptoHash::default() { - self.chain_store_update.get_genesis_height() - } else { - self.chain_store_update.get_block_header(last_final_block)?.height() - }; - - let epoch_manager_update = self - .epoch_manager - .add_validator_proposals(BlockHeaderInfo::new(block.header(), last_finalized_height))?; - self.chain_store_update.merge(epoch_manager_update); - - #[cfg(feature = "new_epoch_sync")] - { - // BlockInfo should be already recorded in epoch_manager cache because of `add_validator_proposals` call - self.save_epoch_sync_info_if_finalised(block.header())?; - } - - // Add validated block to the db, even if it's not the canonical fork. - self.chain_store_update.save_block(block.clone()); - self.chain_store_update.inc_block_refcount(prev_hash)?; - - // Save receipt_id_to_shard_id for all outgoing receipts generated in this block - self.save_receipt_id_to_shard_id_for_block( - me.as_ref(), - block.hash(), - prev_hash, - &shard_ids, - )?; - - // Update the chain head if it's the new tip - let res = self.update_head(block.header())?; - - if res.is_some() { - // On the epoch switch record the epoch light client block - // Note that we only do it if `res.is_some()`, i.e. if the current block is the head. - // This is necessary because the computation of the light client block relies on - // `ColNextBlockHash`-es populated, and they are only populated for the canonical - // chain. We need to be careful to avoid a situation when the first block of the epoch - // never becomes a tip of the canonical chain. - // Presently the epoch boundary is defined by the height, and the fork choice rule - // is also just height, so the very first block to cross the epoch end is guaranteed - // to be the head of the chain, and result in the light client block produced. - let prev = self.chain_store_update.get_previous_header(block.header())?; - let prev_epoch_id = prev.epoch_id().clone(); - if block.header().epoch_id() != &prev_epoch_id { - if prev.last_final_block() != &CryptoHash::default() { - let light_client_block = self.create_light_client_block(&prev)?; - self.chain_store_update - .save_epoch_light_client_block(&prev_epoch_id.0, light_client_block); - } - } - - let shard_layout = self.epoch_manager.get_shard_layout_from_prev_block(prev.hash())?; - SHARD_LAYOUT_VERSION.set(shard_layout.version() as i64); - SHARD_LAYOUT_NUM_SHARDS.set(shard_layout.shard_ids().count() as i64); - } - Ok(res) - } - - pub fn create_light_client_block( - &mut self, - header: &BlockHeader, - ) -> Result { - // First update the last next_block, since it might not be set yet - self.chain_store_update.save_next_block_hash(header.prev_hash(), *header.hash()); - - Chain::create_light_client_block( - header, - self.epoch_manager.as_ref(), - &mut self.chain_store_update, - ) - } - - #[allow(dead_code)] - fn verify_orphan_header_approvals(&mut self, header: &BlockHeader) -> Result<(), Error> { - let prev_hash = header.prev_hash(); - let prev_height = match header.prev_height() { - None => { - // this will accept orphans of V1 and V2 - // TODO: reject header V1 and V2 after a certain height - return Ok(()); - } - Some(prev_height) => prev_height, - }; - let height = header.height(); - let epoch_id = header.epoch_id(); - let approvals = header.approvals(); - self.epoch_manager.verify_approvals_and_threshold_orphan( - epoch_id, - &|approvals, stakes| { - Doomslug::can_approved_block_be_produced( - self.doomslug_threshold_mode, - approvals, - stakes, - ) - }, - prev_hash, - prev_height, - height, - approvals, - ) - } - - /// Update the header head if this header has most work. - fn update_header_head_if_not_challenged( - &mut self, - header: &BlockHeader, - ) -> Result, Error> { - let header_head = self.chain_store_update.header_head()?; - if header.height() > header_head.height { - let tip = Tip::from_header(header); - self.chain_store_update.save_header_head_if_not_challenged(&tip)?; - debug!(target: "chain", "Header head updated to {} at {}", tip.last_block_hash, tip.height); - metrics::HEADER_HEAD_HEIGHT.set(tip.height as i64); - - Ok(Some(tip)) - } else { - Ok(None) - } - } - - fn update_final_head_from_block(&mut self, header: &BlockHeader) -> Result, Error> { - let final_head = self.chain_store_update.final_head()?; - let last_final_block_header = - match self.chain_store_update.get_block_header(header.last_final_block()) { - Ok(final_header) => final_header, - Err(Error::DBNotFoundErr(_)) => return Ok(None), - Err(err) => return Err(err), - }; - if last_final_block_header.height() > final_head.height { - let tip = Tip::from_header(&last_final_block_header); - self.chain_store_update.save_final_head(&tip)?; - Ok(Some(tip)) - } else { - Ok(None) - } - } - - /// Directly updates the head if we've just appended a new block to it or handle - /// the situation where the block has higher height to have a fork - fn update_head(&mut self, header: &BlockHeader) -> Result, Error> { - // if we made a fork with higher height than the head (which should also be true - // when extending the head), update it - self.update_final_head_from_block(header)?; - let head = self.chain_store_update.head()?; - if header.height() > head.height { - let tip = Tip::from_header(header); - - self.chain_store_update.save_body_head(&tip)?; - metrics::BLOCK_HEIGHT_HEAD.set(tip.height as i64); - metrics::BLOCK_ORDINAL_HEAD.set(header.block_ordinal() as i64); - debug!(target: "chain", "Head updated to {} at {}", tip.last_block_hash, tip.height); - Ok(Some(tip)) - } else { - Ok(None) - } - } - - /// Marks a block as invalid, - fn mark_block_as_challenged( - &mut self, - block_hash: &CryptoHash, - challenger_hash: Option<&CryptoHash>, - ) -> Result<(), Error> { - info!(target: "chain", "Marking {} as challenged block (challenged in {:?}) and updating the chain.", block_hash, challenger_hash); - let block_header = match self.chain_store_update.get_block_header(block_hash) { - Ok(block_header) => block_header, - Err(e) => match e { - Error::DBNotFoundErr(_) => { - // The block wasn't seen yet, still challenge is good. - self.chain_store_update.save_challenged_block(*block_hash); - return Ok(()); - } - _ => return Err(e), - }, - }; - - let cur_block_at_same_height = - match self.chain_store_update.get_block_hash_by_height(block_header.height()) { - Ok(bh) => Some(bh), - Err(e) => match e { - Error::DBNotFoundErr(_) => None, - _ => return Err(e), - }, - }; - - self.chain_store_update.save_challenged_block(*block_hash); - - // If the block being invalidated is on the canonical chain, update head - if cur_block_at_same_height == Some(*block_hash) { - // We only consider two candidates for the new head: the challenger and the block - // immediately preceding the block being challenged - // It could be that there is a better chain known. However, it is extremely unlikely, - // and even if there's such chain available, the very next block built on it will - // bring this node's head to that chain. - let prev_header = self.chain_store_update.get_block_header(block_header.prev_hash())?; - let prev_height = prev_header.height(); - let new_head_header = if let Some(hash) = challenger_hash { - let challenger_header = self.chain_store_update.get_block_header(hash)?; - if challenger_header.height() > prev_height { - challenger_header - } else { - prev_header - } - } else { - prev_header - }; - let last_final_block = *new_head_header.last_final_block(); - - let tip = Tip::from_header(&new_head_header); - self.chain_store_update.save_head(&tip)?; - let new_final_header = self.chain_store_update.get_block_header(&last_final_block)?; - self.chain_store_update.save_final_head(&Tip::from_header(&new_final_header))?; - } - - Ok(()) - } - - pub fn set_state_finalize( - &mut self, - shard_id: ShardId, - sync_hash: CryptoHash, - shard_state_header: ShardStateSyncResponseHeader, - ) -> Result<(), Error> { - let _span = - tracing::debug_span!(target: "sync", "chain_update_set_state_finalize").entered(); - let (chunk, incoming_receipts_proofs) = match shard_state_header { - ShardStateSyncResponseHeader::V1(shard_state_header) => ( - ShardChunk::V1(shard_state_header.chunk), - shard_state_header.incoming_receipts_proofs, - ), - ShardStateSyncResponseHeader::V2(shard_state_header) => { - (shard_state_header.chunk, shard_state_header.incoming_receipts_proofs) - } - }; - - let block_header = self - .chain_store_update - .get_block_header_on_chain_by_height(&sync_hash, chunk.height_included())?; - - // Getting actual incoming receipts. - let mut receipt_proof_response: Vec = vec![]; - for incoming_receipt_proof in incoming_receipts_proofs.iter() { - let ReceiptProofResponse(hash, _) = incoming_receipt_proof; - let block_header = self.chain_store_update.get_block_header(hash)?; - if block_header.height() <= chunk.height_included() { - receipt_proof_response.push(incoming_receipt_proof.clone()); - } - } - let receipts = collect_receipts_from_response(&receipt_proof_response); - // Prev block header should be present during state sync, since headers have been synced at this point. - let gas_price = if block_header.height() == self.chain_store_update.get_genesis_height() { - block_header.next_gas_price() - } else { - self.chain_store_update.get_block_header(block_header.prev_hash())?.next_gas_price() - }; - - let chunk_header = chunk.cloned_header(); - let gas_limit = chunk_header.gas_limit(); - // This is set to false because the value is only relevant - // during protocol version RestoreReceiptsAfterFixApplyChunks. - // TODO(nikurt): Determine the value correctly. - let is_first_block_with_chunk_of_version = false; - - let apply_result = self.runtime_adapter.apply_transactions( - RuntimeStorageConfig::new(chunk_header.prev_state_root(), true), - ApplyTransactionsChunkContext { - shard_id, - gas_limit, - last_validator_proposals: chunk_header.prev_validator_proposals(), - is_first_block_with_chunk_of_version, - is_new_chunk: true, - }, - ApplyTransactionsBlockContext { - height: chunk_header.height_included(), - block_hash: *block_header.hash(), - prev_block_hash: *chunk_header.prev_block_hash(), - block_timestamp: block_header.raw_timestamp(), - gas_price, - challenges_result: block_header.challenges_result().clone(), - random_seed: *block_header.random_value(), - }, - &receipts, - chunk.transactions(), - )?; - - let (outcome_root, outcome_proofs) = - ApplyTransactionResult::compute_outcomes_proof(&apply_result.outcomes); - - self.chain_store_update.save_chunk(chunk); - - let shard_uid = self.epoch_manager.shard_id_to_uid(shard_id, block_header.epoch_id())?; - let flat_storage_manager = self.runtime_adapter.get_flat_storage_manager(); - let store_update = flat_storage_manager.save_flat_state_changes( - *block_header.hash(), - *chunk_header.prev_block_hash(), - chunk_header.height_included(), - shard_uid, - apply_result.trie_changes.state_changes(), - )?; - self.chain_store_update.merge(store_update); - - self.chain_store_update.save_trie_changes(apply_result.trie_changes); - let chunk_extra = ChunkExtra::new( - &apply_result.new_root, - outcome_root, - apply_result.validator_proposals, - apply_result.total_gas_burnt, - gas_limit, - apply_result.total_balance_burnt, - ); - self.chain_store_update.save_chunk_extra(block_header.hash(), &shard_uid, chunk_extra); - - self.chain_store_update.save_outgoing_receipt( - block_header.hash(), - shard_id, - apply_result.outgoing_receipts, - ); - // Saving transaction results. - self.chain_store_update.save_outcomes_with_proofs( - block_header.hash(), - shard_id, - apply_result.outcomes, - outcome_proofs, - ); - // Saving all incoming receipts. - for receipt_proof_response in incoming_receipts_proofs { - self.chain_store_update.save_incoming_receipt( - &receipt_proof_response.0, - shard_id, - receipt_proof_response.1, - ); - } - Ok(()) - } - - pub fn set_state_finalize_on_height( - &mut self, - height: BlockHeight, - shard_id: ShardId, - sync_hash: CryptoHash, - ) -> Result { - let _span = tracing::debug_span!(target: "sync", "set_state_finalize_on_height").entered(); - let block_header_result = - self.chain_store_update.get_block_header_on_chain_by_height(&sync_hash, height); - if let Err(_) = block_header_result { - // No such height, go ahead. - return Ok(true); - } - let block_header = block_header_result?; - if block_header.hash() == &sync_hash { - // Don't continue - return Ok(false); - } - let prev_block_header = - self.chain_store_update.get_block_header(block_header.prev_hash())?; - - let shard_uid = self.epoch_manager.shard_id_to_uid(shard_id, block_header.epoch_id())?; - let chunk_extra = - self.chain_store_update.get_chunk_extra(prev_block_header.hash(), &shard_uid)?; - - let apply_result = self.runtime_adapter.apply_transactions( - RuntimeStorageConfig::new(*chunk_extra.state_root(), true), - ApplyTransactionsChunkContext { - shard_id, - last_validator_proposals: chunk_extra.validator_proposals(), - gas_limit: chunk_extra.gas_limit(), - is_new_chunk: false, - is_first_block_with_chunk_of_version: false, - }, - ApplyTransactionsBlockContext::from_header( - &block_header, - prev_block_header.next_gas_price(), - ), - &[], - &[], - )?; - let flat_storage_manager = self.runtime_adapter.get_flat_storage_manager(); - let store_update = flat_storage_manager.save_flat_state_changes( - *block_header.hash(), - *prev_block_header.hash(), - height, - shard_uid, - apply_result.trie_changes.state_changes(), - )?; - self.chain_store_update.merge(store_update); - self.chain_store_update.save_trie_changes(apply_result.trie_changes); - - let mut new_chunk_extra = ChunkExtra::clone(&chunk_extra); - *new_chunk_extra.state_root_mut() = apply_result.new_root; - - self.chain_store_update.save_chunk_extra(block_header.hash(), &shard_uid, new_chunk_extra); - Ok(true) - } -} - -/// Epoch sync specific functions. -#[cfg(feature = "new_epoch_sync")] -impl<'a> ChainUpdate<'a> { - /// This function assumes `BlockInfo` is already retrievable from `epoch_manager`. - /// This can be achieved by calling `add_validator_proposals`. - fn save_epoch_sync_info_if_finalised(&mut self, header: &BlockHeader) -> Result<(), Error> { - let block_info = self.epoch_manager.get_block_info(header.hash())?; - let epoch_first_block_hash = block_info.epoch_first_block(); - - if *epoch_first_block_hash == CryptoHash::default() { - // This is the genesis epoch. We don't have any fully finalised epoch yet. - return Ok(()); - } - - let epoch_first_block_info = self.epoch_manager.get_block_info(epoch_first_block_hash)?; - let prev_epoch_last_block_hash = epoch_first_block_info.prev_hash(); - - if *prev_epoch_last_block_hash == CryptoHash::default() { - // This is the genesis epoch. We don't have any fully finalised epoch yet. - return Ok(()); - } - let prev_epoch_last_block_info = - self.epoch_manager.get_block_info(prev_epoch_last_block_hash)?; - - if prev_epoch_last_block_info.epoch_id() == epoch_first_block_info.epoch_id() { - // Previous epoch is the genesis epoch. We don't have any fully finalised epoch yet. - return Ok(()); - } - - // Check that last finalised block is after epoch first block. - // So, that it is in the current epoch. - let last_final_block_hash = header.last_final_block(); - if *last_final_block_hash == CryptoHash::default() { - // We didn't finalise any blocks yet. We don't have any fully finalised epoch yet. - return Ok(()); - } - let last_final_block_info = self.epoch_manager.get_block_info(last_final_block_hash)?; - if last_final_block_info.epoch_id() != epoch_first_block_info.epoch_id() { - // Last finalised block is in the previous epoch. - // We didn't finalise header with `epoch_sync_data_hash` for the previous epoch yet. - return Ok(()); - } - if self - .chain_store_update - .store() - .exists(DBCol::EpochSyncInfo, prev_epoch_last_block_info.epoch_id().as_ref())? - { - // We already wrote `EpochSyncInfo` for this epoch. - // Probably during epoch sync. - return Ok(()); - } - self.save_epoch_sync_info_impl(&prev_epoch_last_block_info, epoch_first_block_hash) - } - - /// If the block is the last one in the epoch - /// construct and record `EpochSyncInfo` to `self.chain_store_update`. - fn save_epoch_sync_info_impl( - &mut self, - last_block_info: &BlockInfo, - next_epoch_first_hash: &CryptoHash, - ) -> Result<(), Error> { - let mut store_update = self.chain_store_update.store().store_update(); - store_update - .set_ser( - DBCol::EpochSyncInfo, - last_block_info.epoch_id().as_ref(), - &self.create_epoch_sync_info(last_block_info, next_epoch_first_hash, None)?, - ) - .map_err(EpochError::from)?; - self.chain_store_update.merge(store_update); - Ok(()) - } - - /// Create a pair of `BlockHeader`s necessary to create `BlockInfo` for `block_hash`: - /// - header for `block_hash` - /// - header for `last_final_block` of `block_hash` header - fn get_header_pair( - &self, - block_hash: &CryptoHash, - ) -> Result<(BlockHeader, BlockHeader), Error> { - let header = self.chain_store_update.get_block_header(block_hash)?; - // `block_hash` can correspond to genesis block, for which there is no last final block recorded, - // because `last_final_block` for genesis is `CryptoHash::default()` - // Here we return just the same genesis block header as last known block header - // TODO(posvyatokum) process this case carefully in epoch sync validation - // TODO(posvyatokum) process this carefully in saving the parts of epoch sync data - let last_finalised_header = { - if *header.last_final_block() == CryptoHash::default() { - header.clone() - } else { - self.chain_store_update.get_block_header(header.last_final_block())? - } - }; - Ok((header, last_finalised_header)) - } - - /// For epoch sync we need to save: - /// - (*) first header of the next epoch (contains `epoch_sync_data_hash` for `EpochInfo` validation) - /// - first header of the epoch - /// - last header of the epoch - /// - prev last header of the epoch - /// - every header on chain from `last_final_block` to the end of the epoch - /// - (*) header of the `last_final_block` for each of previously mentioned headers - /// - /// Because headers may repeat between those points, we use one `HashMap` to store them indexed by hash. - /// - /// Headers not marked with (*) need to be saved on the syncing node. - /// Headers marked with (*) only needed for `EpochSyncInfo` validation. - fn get_epoch_sync_info_headers( - &self, - last_block_info: &BlockInfo, - next_epoch_first_hash: &CryptoHash, - ) -> Result<(HashMap, HashSet), Error> { - let mut headers = HashMap::new(); - let mut headers_to_save = HashSet::new(); - - let mut add_header = |block_hash: &CryptoHash| -> Result<(), Error> { - let (header, last_finalised_header) = self.get_header_pair(block_hash)?; - headers.insert(*header.hash(), header); - headers.insert(*last_finalised_header.hash(), last_finalised_header); - headers_to_save.insert(*block_hash); - Ok(()) - }; - - add_header(next_epoch_first_hash)?; - add_header(last_block_info.epoch_first_block())?; - add_header(last_block_info.hash())?; - add_header(last_block_info.prev_hash())?; - - // If we didn't add `last_final_block_hash` yet, go down the chain until we find it. - if last_block_info.hash() != last_block_info.last_final_block_hash() - && last_block_info.prev_hash() != last_block_info.last_final_block_hash() - { - let mut current_header = - self.chain_store_update.get_block_header(last_block_info.prev_hash())?; - while current_header.hash() != last_block_info.last_final_block_hash() { - // This only should happen if BlockInfo data is incorrect. - // Without this assert same BlockInfo will cause infinite loop instead of crash with a message. - assert!( - current_header.height() > last_block_info.last_finalized_height(), - "Reached block at height {:?} with hash {:?} from {:?}", - current_header.height(), - current_header.hash(), - last_block_info - ); - - // current_header was already added, as we start from current_header = prev_header. - current_header = - self.chain_store_update.get_block_header(current_header.prev_hash())?; - add_header(current_header.hash())?; - } - } - - // We don't need to save `next_epoch_first_hash` during `EpochSyncInfo` processing. - // It is only needed for validation. - headers_to_save.remove(next_epoch_first_hash); - - Ok((headers, headers_to_save)) - } - - /// Data that is necessary to prove Epoch in new Epoch Sync. - pub fn create_epoch_sync_info( - &self, - last_block_info: &BlockInfo, - next_epoch_first_hash: &CryptoHash, - hash_to_prev_hash: Option<&HashMap>, - ) -> Result { - let mut all_block_hashes = - self.epoch_manager.get_all_epoch_hashes(last_block_info, hash_to_prev_hash)?; - all_block_hashes.reverse(); - - let (headers, headers_to_save) = - self.get_epoch_sync_info_headers(last_block_info, next_epoch_first_hash)?; - - let epoch_id = last_block_info.epoch_id(); - let next_epoch_id = self.epoch_manager.get_next_epoch_id(last_block_info.hash())?; - let next_next_epoch_id = EpochId(*last_block_info.hash()); - - Ok(EpochSyncInfo { - all_block_hashes, - headers, - headers_to_save, - next_epoch_first_hash: *next_epoch_first_hash, - epoch_info: (*self.epoch_manager.get_epoch_info(epoch_id)?).clone(), - next_epoch_info: (*self.epoch_manager.get_epoch_info(&next_epoch_id)?).clone(), - next_next_epoch_info: (*self.epoch_manager.get_epoch_info(&next_next_epoch_id)?) - .clone(), - }) - } -} - pub fn do_apply_chunks( block_hash: CryptoHash, block_height: BlockHeight, diff --git a/chain/chain/src/chain_update.rs b/chain/chain/src/chain_update.rs new file mode 100644 index 00000000000..84fde664c07 --- /dev/null +++ b/chain/chain/src/chain_update.rs @@ -0,0 +1,1057 @@ +use crate::block_processing_utils::BlockPreprocessInfo; +use crate::chain::collect_receipts_from_response; +use crate::metrics::{SHARD_LAYOUT_NUM_SHARDS, SHARD_LAYOUT_VERSION}; +use crate::store::{ChainStore, ChainStoreAccess, ChainStoreUpdate}; + +use crate::types::{ + ApplySplitStateResultOrStateChanges, ApplyTransactionResult, ApplyTransactionsBlockContext, + ApplyTransactionsChunkContext, RuntimeAdapter, RuntimeStorageConfig, +}; +use crate::update_shard::{ + NewChunkResult, OldChunkResult, ShardBlockUpdateResult, ShardUpdateResult, StateSplitResult, +}; +use crate::{metrics, DoomslugThresholdMode}; +use crate::{Chain, Doomslug}; +use near_chain_primitives::error::Error; +use near_epoch_manager::shard_tracker::ShardTracker; +use near_epoch_manager::types::BlockHeaderInfo; +use near_epoch_manager::EpochManagerAdapter; +use near_primitives::block::{Block, Tip}; +use near_primitives::block_header::BlockHeader; +#[cfg(feature = "new_epoch_sync")] +use near_primitives::epoch_manager::{block_info::BlockInfo, epoch_sync::EpochSyncInfo}; +use near_primitives::hash::CryptoHash; +use near_primitives::shard_layout::{account_id_to_shard_id, account_id_to_shard_uid, ShardUId}; +use near_primitives::sharding::ShardChunk; +use near_primitives::state_sync::{ReceiptProofResponse, ShardStateSyncResponseHeader}; +use near_primitives::types::chunk_extra::ChunkExtra; +use near_primitives::types::{ + AccountId, BlockExtra, BlockHeight, BlockHeightDelta, NumShards, ShardId, +}; +use near_primitives::views::LightClientBlockView; +use std::collections::HashMap; +#[cfg(feature = "new_epoch_sync")] +use std::collections::HashSet; +use std::sync::Arc; +use tracing::{debug, info, warn}; + +/// Chain update helper, contains information that is needed to process block +/// and decide to accept it or reject it. +/// If rejected nothing will be updated in underlying storage. +/// Safe to stop process mid way (Ctrl+C or crash). +pub struct ChainUpdate<'a> { + pub(crate) epoch_manager: Arc, + shard_tracker: ShardTracker, + runtime_adapter: Arc, + pub(crate) chain_store_update: ChainStoreUpdate<'a>, + doomslug_threshold_mode: DoomslugThresholdMode, + #[allow(unused)] + transaction_validity_period: BlockHeightDelta, +} + +impl<'a> ChainUpdate<'a> { + pub fn new( + store: &'a mut ChainStore, + epoch_manager: Arc, + shard_tracker: ShardTracker, + runtime_adapter: Arc, + doomslug_threshold_mode: DoomslugThresholdMode, + transaction_validity_period: BlockHeightDelta, + ) -> Self { + let chain_store_update: ChainStoreUpdate<'_> = store.store_update(); + Self::new_impl( + epoch_manager, + shard_tracker, + runtime_adapter, + doomslug_threshold_mode, + transaction_validity_period, + chain_store_update, + ) + } + + fn new_impl( + epoch_manager: Arc, + shard_tracker: ShardTracker, + runtime_adapter: Arc, + doomslug_threshold_mode: DoomslugThresholdMode, + transaction_validity_period: BlockHeightDelta, + chain_store_update: ChainStoreUpdate<'a>, + ) -> Self { + ChainUpdate { + epoch_manager, + shard_tracker, + runtime_adapter, + chain_store_update, + doomslug_threshold_mode, + transaction_validity_period, + } + } + + /// Commit changes to the chain into the database. + pub fn commit(self) -> Result<(), Error> { + self.chain_store_update.commit() + } + + /// For all the outgoing receipts generated in block `hash` at the shards we + /// are tracking in this epoch, save a mapping from receipt ids to the + /// destination shard ids that the receipt will be sent to in the next + /// block. + /// + /// Note that this function should be called after `save_block` is called on + /// this block because it requires that the block info is available in + /// EpochManager, otherwise it will return an error. + fn save_receipt_id_to_shard_id_for_block( + &mut self, + account_id: Option<&AccountId>, + hash: &CryptoHash, + prev_hash: &CryptoHash, + shard_ids: &[ShardId], + ) -> Result<(), Error> { + let mut list = vec![]; + for &shard_id in shard_ids { + if self.shard_tracker.care_about_shard(account_id, prev_hash, shard_id, true) { + list.push(self.get_receipt_id_to_shard_id(hash, shard_id)?); + } + } + for map in list { + for (receipt_id, shard_id) in map { + self.chain_store_update.save_receipt_id_to_shard_id(receipt_id, shard_id); + } + } + Ok(()) + } + + /// Returns a mapping from the receipt id to the destination shard id. + fn get_receipt_id_to_shard_id( + &mut self, + hash: &CryptoHash, + shard_id: u64, + ) -> Result, Error> { + let outgoing_receipts = self.chain_store_update.get_outgoing_receipts(hash, shard_id); + let outgoing_receipts = if let Ok(outgoing_receipts) = outgoing_receipts { + outgoing_receipts + } else { + return Ok(HashMap::new()); + }; + let shard_layout = self.epoch_manager.get_shard_layout_from_prev_block(hash)?; + let outgoing_receipts = outgoing_receipts + .iter() + .map(|receipt| { + (receipt.receipt_id, account_id_to_shard_id(&receipt.receiver_id, &shard_layout)) + }) + .collect(); + Ok(outgoing_receipts) + } + + pub(crate) fn apply_chunk_postprocessing( + &mut self, + block: &Block, + apply_results: Vec, + ) -> Result<(), Error> { + let _span = tracing::debug_span!(target: "chain", "apply_chunk_postprocessing").entered(); + for result in apply_results { + match result { + ShardUpdateResult::Stateful(result) => { + self.process_apply_chunk_result(block, result)? + } + ShardUpdateResult::Stateless(results) => { + for (block_hash, shard_uid, chunk_extra) in results { + let expected_chunk_extra = + self.chain_store_update.get_chunk_extra(&block_hash, &shard_uid)?; + assert_eq!( + &chunk_extra, + expected_chunk_extra.as_ref(), + "For stateless validation, chunk extras for block {} and shard {} do not match", + block_hash, + shard_uid + ); + } + } + } + } + Ok(()) + } + + /// Postprocess split state results or state changes, do the necessary update on chain + /// for split state results: store the chunk extras and trie changes for the split states + /// for state changes, store the state changes for splitting states + fn process_split_state( + &mut self, + block: &Block, + shard_uid: &ShardUId, + apply_results_or_state_changes: ApplySplitStateResultOrStateChanges, + ) -> Result<(), Error> { + let block_hash = block.hash(); + let prev_hash = block.header().prev_hash(); + let height = block.header().height(); + match apply_results_or_state_changes { + ApplySplitStateResultOrStateChanges::ApplySplitStateResults(mut results) => { + tracing::debug!(target: "resharding", height, ?shard_uid, "process_split_state apply"); + + // Sort the results so that the gas reassignment is deterministic. + results.sort_unstable_by_key(|r| r.shard_uid); + // Drop the mutability as we no longer need it. + let results = results; + + // Split validator_proposals, gas_burnt, balance_burnt to each split shard + // and store the chunk extra for split shards + // Note that here we do not split outcomes by the new shard layout, we simply store + // the outcome_root from the parent shard. This is because outcome proofs are + // generated per shard using the old shard layout and stored in the database. + // For these proofs to work, we must store the outcome root per shard + // using the old shard layout instead of the new shard layout + let chunk_extra = self.chain_store_update.get_chunk_extra(block_hash, shard_uid)?; + let next_epoch_shard_layout = { + let epoch_id = + self.epoch_manager.get_next_epoch_id_from_prev_block(prev_hash)?; + self.epoch_manager.get_shard_layout(&epoch_id)? + }; + + let mut validator_proposals_by_shard: HashMap<_, Vec<_>> = HashMap::new(); + for validator_proposal in chunk_extra.validator_proposals() { + let shard_uid = account_id_to_shard_uid( + validator_proposal.account_id(), + &next_epoch_shard_layout, + ); + validator_proposals_by_shard + .entry(shard_uid) + .or_default() + .push(validator_proposal); + } + + let num_split_shards = next_epoch_shard_layout + .get_split_shard_uids(shard_uid.shard_id()) + .unwrap_or_else(|| panic!("invalid shard layout {:?}", next_epoch_shard_layout)) + .len() as NumShards; + + let total_gas_used = chunk_extra.gas_used(); + let total_balance_burnt = chunk_extra.balance_burnt(); + + // The gas remainder, the split shards will be reassigned one + // unit each until its depleted. + let mut gas_res = total_gas_used % num_split_shards; + // The gas quotient, the split shards will be reassigned the + // full value each. + let gas_split = total_gas_used / num_split_shards; + + // The balance remainder, the split shards will be reassigned one + // unit each until its depleted. + let mut balance_res = (total_balance_burnt % num_split_shards as u128) as NumShards; + // The balance quotient, the split shards will be reassigned the + // full value each. + let balance_split = total_balance_burnt / (num_split_shards as u128); + + let gas_limit = chunk_extra.gas_limit(); + let outcome_root = *chunk_extra.outcome_root(); + + let mut sum_gas_used = 0; + let mut sum_balance_burnt = 0; + + // The gas and balance distribution assumes that we have a result for every split shard. + // TODO(resharding) make sure that is the case. + assert_eq!(num_split_shards, results.len() as u64); + + for result in results { + let gas_burnt = if gas_res > 0 { + gas_res -= 1; + gas_split + 1 + } else { + gas_split + }; + + let balance_burnt = if balance_res > 0 { + balance_res -= 1; + balance_split + 1 + } else { + balance_split + }; + + let new_chunk_extra = ChunkExtra::new( + &result.new_root, + outcome_root, + validator_proposals_by_shard.remove(&result.shard_uid).unwrap_or_default(), + gas_burnt, + gas_limit, + balance_burnt, + ); + sum_gas_used += gas_burnt; + sum_balance_burnt += balance_burnt; + + // TODO(#9430): Support manager.save_flat_state_changes and manager.update_flat_storage_for_shard + // functions to be a part of the same chain_store_update + let flat_storage_manager = self.runtime_adapter.get_flat_storage_manager(); + let store_update = flat_storage_manager.save_flat_state_changes( + *block_hash, + *prev_hash, + block.header().height(), + result.shard_uid, + result.trie_changes.state_changes(), + )?; + flat_storage_manager.update_flat_storage_for_shard(*shard_uid, block)?; + self.chain_store_update.merge(store_update); + + self.chain_store_update.save_chunk_extra( + block_hash, + &result.shard_uid, + new_chunk_extra, + ); + self.chain_store_update.save_trie_changes(result.trie_changes); + } + assert_eq!(sum_gas_used, total_gas_used); + assert_eq!(sum_balance_burnt, total_balance_burnt); + } + ApplySplitStateResultOrStateChanges::StateChangesForSplitStates(state_changes) => { + tracing::debug!(target: "resharding", height, ?shard_uid, "process_split_state store"); + self.chain_store_update.add_state_changes_for_split_states( + *block_hash, + shard_uid.shard_id(), + state_changes, + ); + } + } + Ok(()) + } + + /// Processed results of applying chunk + fn process_apply_chunk_result( + &mut self, + block: &Block, + result: ShardBlockUpdateResult, + ) -> Result<(), Error> { + let block_hash = block.hash(); + let prev_hash = block.header().prev_hash(); + let height = block.header().height(); + match result { + ShardBlockUpdateResult::NewChunk(NewChunkResult { + gas_limit, + shard_uid, + apply_result, + apply_split_result_or_state_changes, + }) => { + let (outcome_root, outcome_paths) = + ApplyTransactionResult::compute_outcomes_proof(&apply_result.outcomes); + let shard_id = shard_uid.shard_id(); + + // Save state root after applying transactions. + self.chain_store_update.save_chunk_extra( + block_hash, + &shard_uid, + ChunkExtra::new( + &apply_result.new_root, + outcome_root, + apply_result.validator_proposals, + apply_result.total_gas_burnt, + gas_limit, + apply_result.total_balance_burnt, + ), + ); + + let flat_storage_manager = self.runtime_adapter.get_flat_storage_manager(); + let store_update = flat_storage_manager.save_flat_state_changes( + *block_hash, + *prev_hash, + height, + shard_uid, + apply_result.trie_changes.state_changes(), + )?; + self.chain_store_update.merge(store_update); + + self.chain_store_update.save_trie_changes(apply_result.trie_changes); + self.chain_store_update.save_outgoing_receipt( + block_hash, + shard_id, + apply_result.outgoing_receipts, + ); + // Save receipt and transaction results. + self.chain_store_update.save_outcomes_with_proofs( + block_hash, + shard_id, + apply_result.outcomes, + outcome_paths, + ); + if let Some(apply_results_or_state_changes) = apply_split_result_or_state_changes { + self.process_split_state(block, &shard_uid, apply_results_or_state_changes)?; + } + } + ShardBlockUpdateResult::OldChunk(OldChunkResult { + shard_uid, + apply_result, + apply_split_result_or_state_changes, + }) => { + let old_extra = self.chain_store_update.get_chunk_extra(prev_hash, &shard_uid)?; + + let mut new_extra = ChunkExtra::clone(&old_extra); + *new_extra.state_root_mut() = apply_result.new_root; + + let flat_storage_manager = self.runtime_adapter.get_flat_storage_manager(); + let store_update = flat_storage_manager.save_flat_state_changes( + *block_hash, + *prev_hash, + height, + shard_uid, + apply_result.trie_changes.state_changes(), + )?; + self.chain_store_update.merge(store_update); + + self.chain_store_update.save_chunk_extra(block_hash, &shard_uid, new_extra); + self.chain_store_update.save_trie_changes(apply_result.trie_changes); + + if let Some(apply_results_or_state_changes) = apply_split_result_or_state_changes { + self.process_split_state(block, &shard_uid, apply_results_or_state_changes)?; + } + } + ShardBlockUpdateResult::StateSplit(StateSplitResult { shard_uid, results }) => { + self.chain_store_update + .remove_state_changes_for_split_states(*block.hash(), shard_uid.shard_id()); + self.process_split_state( + block, + &shard_uid, + ApplySplitStateResultOrStateChanges::ApplySplitStateResults(results), + )?; + } + }; + Ok(()) + } + + /// This is the last step of process_block_single, where we take the preprocess block info + /// apply chunk results and store the results on chain. + pub(crate) fn postprocess_block( + &mut self, + me: &Option, + block: &Block, + block_preprocess_info: BlockPreprocessInfo, + apply_chunks_results: Vec<(ShardId, Result)>, + ) -> Result, Error> { + let shard_ids = self.epoch_manager.shard_ids(block.header().epoch_id())?; + let prev_hash = block.header().prev_hash(); + let results = apply_chunks_results.into_iter().map(|(shard_id, x)| { + if let Err(err) = &x { + warn!(target: "chain", shard_id, hash = %block.hash(), %err, "Error in applying chunk for block"); + } + x + }).collect::, Error>>()?; + self.apply_chunk_postprocessing(block, results)?; + + let BlockPreprocessInfo { + is_caught_up, + state_sync_info, + incoming_receipts, + challenges_result, + challenged_blocks, + .. + } = block_preprocess_info; + + if !is_caught_up { + debug!(target: "chain", %prev_hash, hash = %*block.hash(), "Add block to catch up"); + self.chain_store_update.add_block_to_catchup(*prev_hash, *block.hash()); + } + + for (shard_id, receipt_proofs) in incoming_receipts { + self.chain_store_update.save_incoming_receipt( + block.hash(), + shard_id, + Arc::new(receipt_proofs), + ); + } + if let Some(state_sync_info) = state_sync_info { + self.chain_store_update.add_state_sync_info(state_sync_info); + } + + self.chain_store_update.save_block_extra(block.hash(), BlockExtra { challenges_result }); + for block_hash in challenged_blocks { + self.mark_block_as_challenged(&block_hash, Some(block.hash()))?; + } + + self.chain_store_update.save_block_header(block.header().clone())?; + self.update_header_head_if_not_challenged(block.header())?; + + // If block checks out, record validator proposals for given block. + let last_final_block = block.header().last_final_block(); + let last_finalized_height = if last_final_block == &CryptoHash::default() { + self.chain_store_update.get_genesis_height() + } else { + self.chain_store_update.get_block_header(last_final_block)?.height() + }; + + let epoch_manager_update = self + .epoch_manager + .add_validator_proposals(BlockHeaderInfo::new(block.header(), last_finalized_height))?; + self.chain_store_update.merge(epoch_manager_update); + + #[cfg(feature = "new_epoch_sync")] + { + // BlockInfo should be already recorded in epoch_manager cache because of `add_validator_proposals` call + self.save_epoch_sync_info_if_finalised(block.header())?; + } + + // Add validated block to the db, even if it's not the canonical fork. + self.chain_store_update.save_block(block.clone()); + self.chain_store_update.inc_block_refcount(prev_hash)?; + + // Save receipt_id_to_shard_id for all outgoing receipts generated in this block + self.save_receipt_id_to_shard_id_for_block( + me.as_ref(), + block.hash(), + prev_hash, + &shard_ids, + )?; + + // Update the chain head if it's the new tip + let res = self.update_head(block.header())?; + + if res.is_some() { + // On the epoch switch record the epoch light client block + // Note that we only do it if `res.is_some()`, i.e. if the current block is the head. + // This is necessary because the computation of the light client block relies on + // `ColNextBlockHash`-es populated, and they are only populated for the canonical + // chain. We need to be careful to avoid a situation when the first block of the epoch + // never becomes a tip of the canonical chain. + // Presently the epoch boundary is defined by the height, and the fork choice rule + // is also just height, so the very first block to cross the epoch end is guaranteed + // to be the head of the chain, and result in the light client block produced. + let prev = self.chain_store_update.get_previous_header(block.header())?; + let prev_epoch_id = prev.epoch_id().clone(); + if block.header().epoch_id() != &prev_epoch_id { + if prev.last_final_block() != &CryptoHash::default() { + let light_client_block = self.create_light_client_block(&prev)?; + self.chain_store_update + .save_epoch_light_client_block(&prev_epoch_id.0, light_client_block); + } + } + + let shard_layout = self.epoch_manager.get_shard_layout_from_prev_block(prev.hash())?; + SHARD_LAYOUT_VERSION.set(shard_layout.version() as i64); + SHARD_LAYOUT_NUM_SHARDS.set(shard_layout.shard_ids().count() as i64); + } + Ok(res) + } + + pub fn create_light_client_block( + &mut self, + header: &BlockHeader, + ) -> Result { + // First update the last next_block, since it might not be set yet + self.chain_store_update.save_next_block_hash(header.prev_hash(), *header.hash()); + + Chain::create_light_client_block( + header, + self.epoch_manager.as_ref(), + &mut self.chain_store_update, + ) + } + + #[allow(dead_code)] + fn verify_orphan_header_approvals(&mut self, header: &BlockHeader) -> Result<(), Error> { + let prev_hash = header.prev_hash(); + let prev_height = match header.prev_height() { + None => { + // this will accept orphans of V1 and V2 + // TODO: reject header V1 and V2 after a certain height + return Ok(()); + } + Some(prev_height) => prev_height, + }; + let height = header.height(); + let epoch_id = header.epoch_id(); + let approvals = header.approvals(); + self.epoch_manager.verify_approvals_and_threshold_orphan( + epoch_id, + &|approvals, stakes| { + Doomslug::can_approved_block_be_produced( + self.doomslug_threshold_mode, + approvals, + stakes, + ) + }, + prev_hash, + prev_height, + height, + approvals, + ) + } + + /// Update the header head if this header has most work. + pub(crate) fn update_header_head_if_not_challenged( + &mut self, + header: &BlockHeader, + ) -> Result, Error> { + let header_head = self.chain_store_update.header_head()?; + if header.height() > header_head.height { + let tip = Tip::from_header(header); + self.chain_store_update.save_header_head_if_not_challenged(&tip)?; + debug!(target: "chain", "Header head updated to {} at {}", tip.last_block_hash, tip.height); + metrics::HEADER_HEAD_HEIGHT.set(tip.height as i64); + + Ok(Some(tip)) + } else { + Ok(None) + } + } + + fn update_final_head_from_block(&mut self, header: &BlockHeader) -> Result, Error> { + let final_head = self.chain_store_update.final_head()?; + let last_final_block_header = + match self.chain_store_update.get_block_header(header.last_final_block()) { + Ok(final_header) => final_header, + Err(Error::DBNotFoundErr(_)) => return Ok(None), + Err(err) => return Err(err), + }; + if last_final_block_header.height() > final_head.height { + let tip = Tip::from_header(&last_final_block_header); + self.chain_store_update.save_final_head(&tip)?; + Ok(Some(tip)) + } else { + Ok(None) + } + } + + /// Directly updates the head if we've just appended a new block to it or handle + /// the situation where the block has higher height to have a fork + fn update_head(&mut self, header: &BlockHeader) -> Result, Error> { + // if we made a fork with higher height than the head (which should also be true + // when extending the head), update it + self.update_final_head_from_block(header)?; + let head = self.chain_store_update.head()?; + if header.height() > head.height { + let tip = Tip::from_header(header); + + self.chain_store_update.save_body_head(&tip)?; + metrics::BLOCK_HEIGHT_HEAD.set(tip.height as i64); + metrics::BLOCK_ORDINAL_HEAD.set(header.block_ordinal() as i64); + debug!(target: "chain", "Head updated to {} at {}", tip.last_block_hash, tip.height); + Ok(Some(tip)) + } else { + Ok(None) + } + } + + /// Marks a block as invalid, + pub(crate) fn mark_block_as_challenged( + &mut self, + block_hash: &CryptoHash, + challenger_hash: Option<&CryptoHash>, + ) -> Result<(), Error> { + info!(target: "chain", "Marking {} as challenged block (challenged in {:?}) and updating the chain.", block_hash, challenger_hash); + let block_header = match self.chain_store_update.get_block_header(block_hash) { + Ok(block_header) => block_header, + Err(e) => match e { + Error::DBNotFoundErr(_) => { + // The block wasn't seen yet, still challenge is good. + self.chain_store_update.save_challenged_block(*block_hash); + return Ok(()); + } + _ => return Err(e), + }, + }; + + let cur_block_at_same_height = + match self.chain_store_update.get_block_hash_by_height(block_header.height()) { + Ok(bh) => Some(bh), + Err(e) => match e { + Error::DBNotFoundErr(_) => None, + _ => return Err(e), + }, + }; + + self.chain_store_update.save_challenged_block(*block_hash); + + // If the block being invalidated is on the canonical chain, update head + if cur_block_at_same_height == Some(*block_hash) { + // We only consider two candidates for the new head: the challenger and the block + // immediately preceding the block being challenged + // It could be that there is a better chain known. However, it is extremely unlikely, + // and even if there's such chain available, the very next block built on it will + // bring this node's head to that chain. + let prev_header = self.chain_store_update.get_block_header(block_header.prev_hash())?; + let prev_height = prev_header.height(); + let new_head_header = if let Some(hash) = challenger_hash { + let challenger_header = self.chain_store_update.get_block_header(hash)?; + if challenger_header.height() > prev_height { + challenger_header + } else { + prev_header + } + } else { + prev_header + }; + let last_final_block = *new_head_header.last_final_block(); + + let tip = Tip::from_header(&new_head_header); + self.chain_store_update.save_head(&tip)?; + let new_final_header = self.chain_store_update.get_block_header(&last_final_block)?; + self.chain_store_update.save_final_head(&Tip::from_header(&new_final_header))?; + } + + Ok(()) + } + + pub fn set_state_finalize( + &mut self, + shard_id: ShardId, + sync_hash: CryptoHash, + shard_state_header: ShardStateSyncResponseHeader, + ) -> Result<(), Error> { + let _span = + tracing::debug_span!(target: "sync", "chain_update_set_state_finalize").entered(); + let (chunk, incoming_receipts_proofs) = match shard_state_header { + ShardStateSyncResponseHeader::V1(shard_state_header) => ( + ShardChunk::V1(shard_state_header.chunk), + shard_state_header.incoming_receipts_proofs, + ), + ShardStateSyncResponseHeader::V2(shard_state_header) => { + (shard_state_header.chunk, shard_state_header.incoming_receipts_proofs) + } + }; + + let block_header = self + .chain_store_update + .get_block_header_on_chain_by_height(&sync_hash, chunk.height_included())?; + + // Getting actual incoming receipts. + let mut receipt_proof_response: Vec = vec![]; + for incoming_receipt_proof in incoming_receipts_proofs.iter() { + let ReceiptProofResponse(hash, _) = incoming_receipt_proof; + let block_header = self.chain_store_update.get_block_header(hash)?; + if block_header.height() <= chunk.height_included() { + receipt_proof_response.push(incoming_receipt_proof.clone()); + } + } + let receipts = collect_receipts_from_response(&receipt_proof_response); + // Prev block header should be present during state sync, since headers have been synced at this point. + let gas_price = if block_header.height() == self.chain_store_update.get_genesis_height() { + block_header.next_gas_price() + } else { + self.chain_store_update.get_block_header(block_header.prev_hash())?.next_gas_price() + }; + + let chunk_header = chunk.cloned_header(); + let gas_limit = chunk_header.gas_limit(); + // This is set to false because the value is only relevant + // during protocol version RestoreReceiptsAfterFixApplyChunks. + // TODO(nikurt): Determine the value correctly. + let is_first_block_with_chunk_of_version = false; + + let apply_result = self.runtime_adapter.apply_transactions( + RuntimeStorageConfig::new(chunk_header.prev_state_root(), true), + ApplyTransactionsChunkContext { + shard_id, + gas_limit, + last_validator_proposals: chunk_header.prev_validator_proposals(), + is_first_block_with_chunk_of_version, + is_new_chunk: true, + }, + ApplyTransactionsBlockContext { + height: chunk_header.height_included(), + block_hash: *block_header.hash(), + prev_block_hash: *chunk_header.prev_block_hash(), + block_timestamp: block_header.raw_timestamp(), + gas_price, + challenges_result: block_header.challenges_result().clone(), + random_seed: *block_header.random_value(), + }, + &receipts, + chunk.transactions(), + )?; + + let (outcome_root, outcome_proofs) = + ApplyTransactionResult::compute_outcomes_proof(&apply_result.outcomes); + + self.chain_store_update.save_chunk(chunk); + + let shard_uid = self.epoch_manager.shard_id_to_uid(shard_id, block_header.epoch_id())?; + let flat_storage_manager = self.runtime_adapter.get_flat_storage_manager(); + let store_update = flat_storage_manager.save_flat_state_changes( + *block_header.hash(), + *chunk_header.prev_block_hash(), + chunk_header.height_included(), + shard_uid, + apply_result.trie_changes.state_changes(), + )?; + self.chain_store_update.merge(store_update); + + self.chain_store_update.save_trie_changes(apply_result.trie_changes); + let chunk_extra = ChunkExtra::new( + &apply_result.new_root, + outcome_root, + apply_result.validator_proposals, + apply_result.total_gas_burnt, + gas_limit, + apply_result.total_balance_burnt, + ); + self.chain_store_update.save_chunk_extra(block_header.hash(), &shard_uid, chunk_extra); + + self.chain_store_update.save_outgoing_receipt( + block_header.hash(), + shard_id, + apply_result.outgoing_receipts, + ); + // Saving transaction results. + self.chain_store_update.save_outcomes_with_proofs( + block_header.hash(), + shard_id, + apply_result.outcomes, + outcome_proofs, + ); + // Saving all incoming receipts. + for receipt_proof_response in incoming_receipts_proofs { + self.chain_store_update.save_incoming_receipt( + &receipt_proof_response.0, + shard_id, + receipt_proof_response.1, + ); + } + Ok(()) + } + + pub fn set_state_finalize_on_height( + &mut self, + height: BlockHeight, + shard_id: ShardId, + sync_hash: CryptoHash, + ) -> Result { + let _span = tracing::debug_span!(target: "sync", "set_state_finalize_on_height").entered(); + let block_header_result = + self.chain_store_update.get_block_header_on_chain_by_height(&sync_hash, height); + if let Err(_) = block_header_result { + // No such height, go ahead. + return Ok(true); + } + let block_header = block_header_result?; + if block_header.hash() == &sync_hash { + // Don't continue + return Ok(false); + } + let prev_block_header = + self.chain_store_update.get_block_header(block_header.prev_hash())?; + + let shard_uid = self.epoch_manager.shard_id_to_uid(shard_id, block_header.epoch_id())?; + let chunk_extra = + self.chain_store_update.get_chunk_extra(prev_block_header.hash(), &shard_uid)?; + + let apply_result = self.runtime_adapter.apply_transactions( + RuntimeStorageConfig::new(*chunk_extra.state_root(), true), + ApplyTransactionsChunkContext { + shard_id, + last_validator_proposals: chunk_extra.validator_proposals(), + gas_limit: chunk_extra.gas_limit(), + is_new_chunk: false, + is_first_block_with_chunk_of_version: false, + }, + ApplyTransactionsBlockContext::from_header( + &block_header, + prev_block_header.next_gas_price(), + ), + &[], + &[], + )?; + let flat_storage_manager = self.runtime_adapter.get_flat_storage_manager(); + let store_update = flat_storage_manager.save_flat_state_changes( + *block_header.hash(), + *prev_block_header.hash(), + height, + shard_uid, + apply_result.trie_changes.state_changes(), + )?; + self.chain_store_update.merge(store_update); + self.chain_store_update.save_trie_changes(apply_result.trie_changes); + + let mut new_chunk_extra = ChunkExtra::clone(&chunk_extra); + *new_chunk_extra.state_root_mut() = apply_result.new_root; + + self.chain_store_update.save_chunk_extra(block_header.hash(), &shard_uid, new_chunk_extra); + Ok(true) + } +} + +/// Epoch sync specific functions. +#[cfg(feature = "new_epoch_sync")] +impl<'a> ChainUpdate<'a> { + /// This function assumes `BlockInfo` is already retrievable from `epoch_manager`. + /// This can be achieved by calling `add_validator_proposals`. + pub(crate) fn save_epoch_sync_info_if_finalised( + &mut self, + header: &BlockHeader, + ) -> Result<(), Error> { + let block_info = self.epoch_manager.get_block_info(header.hash())?; + let epoch_first_block_hash = block_info.epoch_first_block(); + + if *epoch_first_block_hash == CryptoHash::default() { + // This is the genesis epoch. We don't have any fully finalised epoch yet. + return Ok(()); + } + + let epoch_first_block_info = self.epoch_manager.get_block_info(epoch_first_block_hash)?; + let prev_epoch_last_block_hash = epoch_first_block_info.prev_hash(); + + if *prev_epoch_last_block_hash == CryptoHash::default() { + // This is the genesis epoch. We don't have any fully finalised epoch yet. + return Ok(()); + } + let prev_epoch_last_block_info = + self.epoch_manager.get_block_info(prev_epoch_last_block_hash)?; + + if prev_epoch_last_block_info.epoch_id() == epoch_first_block_info.epoch_id() { + // Previous epoch is the genesis epoch. We don't have any fully finalised epoch yet. + return Ok(()); + } + + // Check that last finalised block is after epoch first block. + // So, that it is in the current epoch. + let last_final_block_hash = header.last_final_block(); + if *last_final_block_hash == CryptoHash::default() { + // We didn't finalise any blocks yet. We don't have any fully finalised epoch yet. + return Ok(()); + } + let last_final_block_info = self.epoch_manager.get_block_info(last_final_block_hash)?; + if last_final_block_info.epoch_id() != epoch_first_block_info.epoch_id() { + // Last finalised block is in the previous epoch. + // We didn't finalise header with `epoch_sync_data_hash` for the previous epoch yet. + return Ok(()); + } + if self.chain_store_update.store().exists( + near_store::DBCol::EpochSyncInfo, + prev_epoch_last_block_info.epoch_id().as_ref(), + )? { + // We already wrote `EpochSyncInfo` for this epoch. + // Probably during epoch sync. + return Ok(()); + } + self.save_epoch_sync_info_impl(&prev_epoch_last_block_info, epoch_first_block_hash) + } + + /// If the block is the last one in the epoch + /// construct and record `EpochSyncInfo` to `self.chain_store_update`. + fn save_epoch_sync_info_impl( + &mut self, + last_block_info: &BlockInfo, + next_epoch_first_hash: &CryptoHash, + ) -> Result<(), Error> { + let mut store_update = self.chain_store_update.store().store_update(); + store_update + .set_ser( + near_store::DBCol::EpochSyncInfo, + last_block_info.epoch_id().as_ref(), + &self.create_epoch_sync_info(last_block_info, next_epoch_first_hash, None)?, + ) + .map_err(near_primitives::errors::EpochError::from)?; + self.chain_store_update.merge(store_update); + Ok(()) + } + + /// Create a pair of `BlockHeader`s necessary to create `BlockInfo` for `block_hash`: + /// - header for `block_hash` + /// - header for `last_final_block` of `block_hash` header + fn get_header_pair( + &self, + block_hash: &CryptoHash, + ) -> Result<(BlockHeader, BlockHeader), Error> { + let header = self.chain_store_update.get_block_header(block_hash)?; + // `block_hash` can correspond to genesis block, for which there is no last final block recorded, + // because `last_final_block` for genesis is `CryptoHash::default()` + // Here we return just the same genesis block header as last known block header + // TODO(posvyatokum) process this case carefully in epoch sync validation + // TODO(posvyatokum) process this carefully in saving the parts of epoch sync data + let last_finalised_header = { + if *header.last_final_block() == CryptoHash::default() { + header.clone() + } else { + self.chain_store_update.get_block_header(header.last_final_block())? + } + }; + Ok((header, last_finalised_header)) + } + + /// For epoch sync we need to save: + /// - (*) first header of the next epoch (contains `epoch_sync_data_hash` for `EpochInfo` validation) + /// - first header of the epoch + /// - last header of the epoch + /// - prev last header of the epoch + /// - every header on chain from `last_final_block` to the end of the epoch + /// - (*) header of the `last_final_block` for each of previously mentioned headers + /// + /// Because headers may repeat between those points, we use one `HashMap` to store them indexed by hash. + /// + /// Headers not marked with (*) need to be saved on the syncing node. + /// Headers marked with (*) only needed for `EpochSyncInfo` validation. + fn get_epoch_sync_info_headers( + &self, + last_block_info: &BlockInfo, + next_epoch_first_hash: &CryptoHash, + ) -> Result<(HashMap, HashSet), Error> { + let mut headers = HashMap::new(); + let mut headers_to_save = HashSet::new(); + + let mut add_header = |block_hash: &CryptoHash| -> Result<(), Error> { + let (header, last_finalised_header) = self.get_header_pair(block_hash)?; + headers.insert(*header.hash(), header); + headers.insert(*last_finalised_header.hash(), last_finalised_header); + headers_to_save.insert(*block_hash); + Ok(()) + }; + + add_header(next_epoch_first_hash)?; + add_header(last_block_info.epoch_first_block())?; + add_header(last_block_info.hash())?; + add_header(last_block_info.prev_hash())?; + + // If we didn't add `last_final_block_hash` yet, go down the chain until we find it. + if last_block_info.hash() != last_block_info.last_final_block_hash() + && last_block_info.prev_hash() != last_block_info.last_final_block_hash() + { + let mut current_header = + self.chain_store_update.get_block_header(last_block_info.prev_hash())?; + while current_header.hash() != last_block_info.last_final_block_hash() { + // This only should happen if BlockInfo data is incorrect. + // Without this assert same BlockInfo will cause infinite loop instead of crash with a message. + assert!( + current_header.height() > last_block_info.last_finalized_height(), + "Reached block at height {:?} with hash {:?} from {:?}", + current_header.height(), + current_header.hash(), + last_block_info + ); + + // current_header was already added, as we start from current_header = prev_header. + current_header = + self.chain_store_update.get_block_header(current_header.prev_hash())?; + add_header(current_header.hash())?; + } + } + + // We don't need to save `next_epoch_first_hash` during `EpochSyncInfo` processing. + // It is only needed for validation. + headers_to_save.remove(next_epoch_first_hash); + + Ok((headers, headers_to_save)) + } + + /// Data that is necessary to prove Epoch in new Epoch Sync. + pub fn create_epoch_sync_info( + &self, + last_block_info: &BlockInfo, + next_epoch_first_hash: &CryptoHash, + hash_to_prev_hash: Option<&HashMap>, + ) -> Result { + let mut all_block_hashes = + self.epoch_manager.get_all_epoch_hashes(last_block_info, hash_to_prev_hash)?; + all_block_hashes.reverse(); + + let (headers, headers_to_save) = + self.get_epoch_sync_info_headers(last_block_info, next_epoch_first_hash)?; + + let epoch_id = last_block_info.epoch_id(); + let next_epoch_id = self.epoch_manager.get_next_epoch_id(last_block_info.hash())?; + let next_next_epoch_id = near_primitives::types::EpochId(*last_block_info.hash()); + + Ok(EpochSyncInfo { + all_block_hashes, + headers, + headers_to_save, + next_epoch_first_hash: *next_epoch_first_hash, + epoch_info: (*self.epoch_manager.get_epoch_info(epoch_id)?).clone(), + next_epoch_info: (*self.epoch_manager.get_epoch_info(&next_epoch_id)?).clone(), + next_next_epoch_info: (*self.epoch_manager.get_epoch_info(&next_next_epoch_id)?) + .clone(), + }) + } +} diff --git a/chain/chain/src/lib.rs b/chain/chain/src/lib.rs index 88395957c13..67f2eeadca6 100644 --- a/chain/chain/src/lib.rs +++ b/chain/chain/src/lib.rs @@ -1,5 +1,6 @@ pub use block_processing_utils::{BlockProcessingArtifact, DoneApplyChunkCallback}; -pub use chain::{check_known, collect_receipts, Chain, ChainUpdate}; +pub use chain::{check_known, collect_receipts, Chain}; +pub use chain_update::ChainUpdate; pub use doomslug::{Doomslug, DoomslugBlockProductionReadiness, DoomslugThresholdMode}; pub use lightclient::{create_light_client_block_view, get_epoch_block_producers_view}; pub use near_chain_primitives::{self, Error}; @@ -11,6 +12,7 @@ pub use types::{Block, BlockHeader, BlockStatus, ChainGenesis, Provenance}; mod block_processing_utils; pub mod blocks_delay_tracker; pub mod chain; +mod chain_update; pub mod chunks_store; pub mod crypto_hash_timer; mod doomslug;