Refactor bank_forks_utils::load() and some snapshot logic (#17492)
Refactor a few functions that are on the load-from-snapshot path, to facilitate adding in incremental snapshots more easily. Additionally, add some tests and doc comments.
This commit is contained in:
@ -9,7 +9,7 @@ use crate::{
|
||||
};
|
||||
use log::*;
|
||||
use solana_runtime::{
|
||||
bank_forks::{BankForks, SnapshotConfig},
|
||||
bank_forks::{ArchiveFormat, BankForks, SnapshotConfig},
|
||||
snapshot_utils,
|
||||
};
|
||||
use solana_sdk::{clock::Slot, genesis_config::GenesisConfig, hash::Hash};
|
||||
@ -21,14 +21,18 @@ pub type LoadResult = result::Result<
|
||||
>;
|
||||
|
||||
fn to_loadresult(
|
||||
brp: BlockstoreProcessorResult,
|
||||
snapshot_hash: Option<(Slot, Hash)>,
|
||||
bpr: BlockstoreProcessorResult,
|
||||
snapshot_slot_and_hash: Option<(Slot, Hash)>,
|
||||
) -> LoadResult {
|
||||
brp.map(|(bank_forks, leader_schedule_cache)| {
|
||||
(bank_forks, leader_schedule_cache, snapshot_hash)
|
||||
bpr.map(|(bank_forks, leader_schedule_cache)| {
|
||||
(bank_forks, leader_schedule_cache, snapshot_slot_and_hash)
|
||||
})
|
||||
}
|
||||
|
||||
/// Load the banks and accounts
|
||||
///
|
||||
/// If a snapshot config is given, and a snapshot is found, it will be loaded. Otherwise, load
|
||||
/// from genesis.
|
||||
pub fn load(
|
||||
genesis_config: &GenesisConfig,
|
||||
blockstore: &Blockstore,
|
||||
@ -48,71 +52,48 @@ pub fn load(
|
||||
fs::create_dir_all(&snapshot_config.snapshot_path)
|
||||
.expect("Couldn't create snapshot directory");
|
||||
|
||||
match snapshot_utils::get_highest_snapshot_archive_path(
|
||||
&snapshot_config.snapshot_package_output_path,
|
||||
) {
|
||||
Some((archive_filename, (archive_slot, archive_snapshot_hash, compression))) => {
|
||||
info!("Loading snapshot package: {:?}", archive_filename);
|
||||
// Fail hard here if snapshot fails to load, don't silently continue
|
||||
|
||||
if account_paths.is_empty() {
|
||||
error!("Account paths not present when booting from snapshot");
|
||||
process::exit(1);
|
||||
}
|
||||
|
||||
let deserialized_bank = snapshot_utils::bank_from_archive(
|
||||
&account_paths,
|
||||
&process_options.frozen_accounts,
|
||||
&snapshot_config.snapshot_path,
|
||||
&archive_filename,
|
||||
compression,
|
||||
genesis_config,
|
||||
process_options.debug_keys.clone(),
|
||||
Some(&crate::builtins::get(process_options.bpf_jit)),
|
||||
process_options.account_indexes.clone(),
|
||||
process_options.accounts_db_caching_enabled,
|
||||
process_options.limit_load_slot_count_from_snapshot,
|
||||
)
|
||||
.expect("Load from snapshot failed");
|
||||
if let Some(shrink_paths) = shrink_paths {
|
||||
deserialized_bank.set_shrink_paths(shrink_paths);
|
||||
}
|
||||
|
||||
let deserialized_snapshot_hash = (
|
||||
deserialized_bank.slot(),
|
||||
deserialized_bank.get_accounts_hash(),
|
||||
);
|
||||
|
||||
if process_options.accounts_db_test_hash_calculation {
|
||||
deserialized_bank.update_accounts_hash_with_index_option(false, true);
|
||||
}
|
||||
|
||||
if deserialized_snapshot_hash != (archive_slot, archive_snapshot_hash) {
|
||||
error!(
|
||||
"Snapshot has mismatch:\narchive: {:?}\ndeserialized: {:?}",
|
||||
archive_snapshot_hash, deserialized_snapshot_hash
|
||||
);
|
||||
process::exit(1);
|
||||
}
|
||||
|
||||
return to_loadresult(
|
||||
blockstore_processor::process_blockstore_from_root(
|
||||
blockstore,
|
||||
deserialized_bank,
|
||||
&process_options,
|
||||
&VerifyRecyclers::default(),
|
||||
transaction_status_sender,
|
||||
cache_block_meta_sender,
|
||||
),
|
||||
Some(deserialized_snapshot_hash),
|
||||
);
|
||||
}
|
||||
None => info!("No snapshot package available"),
|
||||
if let Some((archive_filename, (archive_slot, archive_hash, archive_format))) =
|
||||
snapshot_utils::get_highest_snapshot_archive_path(
|
||||
&snapshot_config.snapshot_package_output_path,
|
||||
)
|
||||
{
|
||||
return load_from_snapshot(
|
||||
&genesis_config,
|
||||
&blockstore,
|
||||
account_paths,
|
||||
shrink_paths,
|
||||
snapshot_config,
|
||||
process_options,
|
||||
transaction_status_sender,
|
||||
cache_block_meta_sender,
|
||||
archive_filename,
|
||||
archive_slot,
|
||||
archive_hash,
|
||||
archive_format,
|
||||
);
|
||||
} else {
|
||||
info!("No snapshot package available; will load from genesis");
|
||||
}
|
||||
} else {
|
||||
info!("Snapshots disabled");
|
||||
info!("Snapshots disabled; will load from genesis");
|
||||
}
|
||||
|
||||
load_from_genesis(
|
||||
&genesis_config,
|
||||
&blockstore,
|
||||
account_paths,
|
||||
process_options,
|
||||
cache_block_meta_sender,
|
||||
)
|
||||
}
|
||||
|
||||
fn load_from_genesis(
|
||||
genesis_config: &GenesisConfig,
|
||||
blockstore: &Blockstore,
|
||||
account_paths: Vec<PathBuf>,
|
||||
process_options: ProcessOptions,
|
||||
cache_block_meta_sender: Option<&CacheBlockMetaSender>,
|
||||
) -> LoadResult {
|
||||
info!("Processing ledger from genesis");
|
||||
to_loadresult(
|
||||
blockstore_processor::process_blockstore(
|
||||
@ -125,3 +106,74 @@ pub fn load(
|
||||
None,
|
||||
)
|
||||
}
|
||||
|
||||
#[allow(clippy::too_many_arguments)]
|
||||
fn load_from_snapshot(
|
||||
genesis_config: &GenesisConfig,
|
||||
blockstore: &Blockstore,
|
||||
account_paths: Vec<PathBuf>,
|
||||
shrink_paths: Option<Vec<PathBuf>>,
|
||||
snapshot_config: &SnapshotConfig,
|
||||
process_options: ProcessOptions,
|
||||
transaction_status_sender: Option<&TransactionStatusSender>,
|
||||
cache_block_meta_sender: Option<&CacheBlockMetaSender>,
|
||||
archive_filename: PathBuf,
|
||||
archive_slot: Slot,
|
||||
archive_hash: Hash,
|
||||
archive_format: ArchiveFormat,
|
||||
) -> LoadResult {
|
||||
info!("Loading snapshot package: {:?}", archive_filename);
|
||||
|
||||
// Fail hard here if snapshot fails to load, don't silently continue
|
||||
if account_paths.is_empty() {
|
||||
error!("Account paths not present when booting from snapshot");
|
||||
process::exit(1);
|
||||
}
|
||||
|
||||
let deserialized_bank = snapshot_utils::bank_from_archive(
|
||||
&account_paths,
|
||||
&process_options.frozen_accounts,
|
||||
&snapshot_config.snapshot_path,
|
||||
&archive_filename,
|
||||
archive_format,
|
||||
genesis_config,
|
||||
process_options.debug_keys.clone(),
|
||||
Some(&crate::builtins::get(process_options.bpf_jit)),
|
||||
process_options.account_indexes.clone(),
|
||||
process_options.accounts_db_caching_enabled,
|
||||
process_options.limit_load_slot_count_from_snapshot,
|
||||
)
|
||||
.expect("Load from snapshot failed");
|
||||
if let Some(shrink_paths) = shrink_paths {
|
||||
deserialized_bank.set_shrink_paths(shrink_paths);
|
||||
}
|
||||
|
||||
if process_options.accounts_db_test_hash_calculation {
|
||||
deserialized_bank.update_accounts_hash_with_index_option(false, true);
|
||||
}
|
||||
|
||||
let deserialized_bank_slot_and_hash = (
|
||||
deserialized_bank.slot(),
|
||||
deserialized_bank.get_accounts_hash(),
|
||||
);
|
||||
|
||||
if deserialized_bank_slot_and_hash != (archive_slot, archive_hash) {
|
||||
error!(
|
||||
"Snapshot has mismatch:\narchive: {:?}\ndeserialized: {:?}",
|
||||
archive_hash, deserialized_bank_slot_and_hash
|
||||
);
|
||||
process::exit(1);
|
||||
}
|
||||
|
||||
to_loadresult(
|
||||
blockstore_processor::process_blockstore_from_root(
|
||||
blockstore,
|
||||
deserialized_bank,
|
||||
&process_options,
|
||||
&VerifyRecyclers::default(),
|
||||
transaction_status_sender,
|
||||
cache_block_meta_sender,
|
||||
),
|
||||
Some(deserialized_bank_slot_and_hash),
|
||||
)
|
||||
}
|
||||
|
Reference in New Issue
Block a user