Remove redundant bounds check from getBlock and getBlockTime (#33901)
JsonRpcRequestProcessor::check_blockstore_root() contained some logic that performed duplicate sanity checking on a Blockstore fetch result. The checking involved creating rocksdb iterators, which has non-trivial overhead. This PR removes the duplicate checking, and also adds comments to help reason about how JsonRpcRequestProcessor interprets the Blockstore result.
This commit is contained in:
parent
73815aee51
commit
03a456e7bb
|
@ -3104,21 +3104,6 @@ impl Blockstore {
|
||||||
matches!(self.db.get::<cf::Root>(slot), Ok(Some(true)))
|
matches!(self.db.get::<cf::Root>(slot), Ok(Some(true)))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns true if a slot is between the rooted slot bounds of the ledger, but has not itself
|
|
||||||
/// been rooted. This is either because the slot was skipped, or due to a gap in ledger data,
|
|
||||||
/// as when booting from a newer snapshot.
|
|
||||||
pub fn is_skipped(&self, slot: Slot) -> bool {
|
|
||||||
let lowest_root = self
|
|
||||||
.rooted_slot_iterator(0)
|
|
||||||
.ok()
|
|
||||||
.and_then(|mut iter| iter.next())
|
|
||||||
.unwrap_or_default();
|
|
||||||
match self.db.get::<cf::Root>(slot).ok().flatten() {
|
|
||||||
Some(_) => false,
|
|
||||||
None => slot < self.max_root() && slot > lowest_root,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn insert_bank_hash(&self, slot: Slot, frozen_hash: Hash, is_duplicate_confirmed: bool) {
|
pub fn insert_bank_hash(&self, slot: Slot, frozen_hash: Hash, is_duplicate_confirmed: bool) {
|
||||||
if let Some(prev_value) = self.bank_hash_cf.get(slot).unwrap() {
|
if let Some(prev_value) = self.bank_hash_cf.get(slot).unwrap() {
|
||||||
if prev_value.frozen_hash() == frozen_hash && prev_value.is_duplicate_confirmed() {
|
if prev_value.frozen_hash() == frozen_hash && prev_value.is_duplicate_confirmed() {
|
||||||
|
@ -6868,22 +6853,6 @@ pub mod tests {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_is_skipped() {
|
|
||||||
let ledger_path = get_tmp_ledger_path_auto_delete!();
|
|
||||||
let blockstore = Blockstore::open(ledger_path.path()).unwrap();
|
|
||||||
let roots = [2, 4, 7, 12, 15];
|
|
||||||
blockstore.set_roots(roots.iter()).unwrap();
|
|
||||||
|
|
||||||
for i in 0..20 {
|
|
||||||
if i < 2 || roots.contains(&i) || i > 15 {
|
|
||||||
assert!(!blockstore.is_skipped(i));
|
|
||||||
} else {
|
|
||||||
assert!(blockstore.is_skipped(i));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_iter_bounds() {
|
fn test_iter_bounds() {
|
||||||
let ledger_path = get_tmp_ledger_path_auto_delete!();
|
let ledger_path = get_tmp_ledger_path_auto_delete!();
|
||||||
|
|
|
@ -1002,26 +1002,37 @@ impl JsonRpcRequestProcessor {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn check_blockstore_root<T>(
|
// Check if the given `slot` is within the blockstore bounds. This function assumes that
|
||||||
|
// `result` is from a blockstore fetch, and that the fetch:
|
||||||
|
// 1) Checked if `slot` is above the lowest cleanup slot (and errored if not)
|
||||||
|
// 2) Checked if `slot` is a root
|
||||||
|
fn check_blockstore_bounds<T>(
|
||||||
&self,
|
&self,
|
||||||
result: &std::result::Result<T, BlockstoreError>,
|
result: &std::result::Result<T, BlockstoreError>,
|
||||||
slot: Slot,
|
slot: Slot,
|
||||||
) -> Result<()> {
|
) -> Result<()> {
|
||||||
if let Err(err) = result {
|
match result {
|
||||||
debug!(
|
// The slot was found, all good
|
||||||
"check_blockstore_root, slot: {:?}, max root: {:?}, err: {:?}",
|
Ok(_) => Ok(()),
|
||||||
slot,
|
// The slot was cleaned up, return Ok() for now to allow fallback to bigtable
|
||||||
self.blockstore.max_root(),
|
Err(BlockstoreError::SlotCleanedUp) => Ok(()),
|
||||||
err
|
// The slot was not cleaned up but also not found
|
||||||
);
|
Err(BlockstoreError::SlotNotRooted) => {
|
||||||
if slot >= self.blockstore.max_root() {
|
let max_root = self.blockstore.max_root();
|
||||||
return Err(RpcCustomError::BlockNotAvailable { slot }.into());
|
debug!("check_blockstore_bounds, slot: {slot}, max root: {max_root}");
|
||||||
}
|
// Our node hasn't seen this slot yet, error out
|
||||||
if self.blockstore.is_skipped(slot) {
|
if slot >= max_root {
|
||||||
return Err(RpcCustomError::SlotSkipped { slot }.into());
|
return Err(RpcCustomError::BlockNotAvailable { slot }.into());
|
||||||
|
}
|
||||||
|
// The slot is within the bounds of the blockstore as the lookup that yielded
|
||||||
|
// `result` checked that `slot` was greater than the blockstore's lowest
|
||||||
|
// cleanup slot and we just checked that `slot` was less than the blockstore's
|
||||||
|
// largest root. Thus, the slot must have been skipped and we can error out.
|
||||||
|
Err(RpcCustomError::SlotSkipped { slot }.into())
|
||||||
}
|
}
|
||||||
|
// Some other Blockstore error, ignore for now
|
||||||
|
_ => Ok(()),
|
||||||
}
|
}
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn check_slot_cleaned_up<T>(
|
fn check_slot_cleaned_up<T>(
|
||||||
|
@ -1098,7 +1109,7 @@ impl JsonRpcRequestProcessor {
|
||||||
{
|
{
|
||||||
self.check_blockstore_writes_complete(slot)?;
|
self.check_blockstore_writes_complete(slot)?;
|
||||||
let result = self.blockstore.get_rooted_block(slot, true);
|
let result = self.blockstore.get_rooted_block(slot, true);
|
||||||
self.check_blockstore_root(&result, slot)?;
|
self.check_blockstore_bounds(&result, slot)?;
|
||||||
let encode_block = |confirmed_block: ConfirmedBlock| -> Result<UiConfirmedBlock> {
|
let encode_block = |confirmed_block: ConfirmedBlock| -> Result<UiConfirmedBlock> {
|
||||||
let mut encoded_block = confirmed_block
|
let mut encoded_block = confirmed_block
|
||||||
.encode_with_options(encoding, encoding_options)
|
.encode_with_options(encoding, encoding_options)
|
||||||
|
@ -1322,7 +1333,7 @@ impl JsonRpcRequestProcessor {
|
||||||
.highest_super_majority_root()
|
.highest_super_majority_root()
|
||||||
{
|
{
|
||||||
let result = self.blockstore.get_rooted_block_time(slot);
|
let result = self.blockstore.get_rooted_block_time(slot);
|
||||||
self.check_blockstore_root(&result, slot)?;
|
self.check_blockstore_bounds(&result, slot)?;
|
||||||
if result.is_err() {
|
if result.is_err() {
|
||||||
if let Some(bigtable_ledger_storage) = &self.bigtable_ledger_storage {
|
if let Some(bigtable_ledger_storage) = &self.bigtable_ledger_storage {
|
||||||
let bigtable_result = bigtable_ledger_storage.get_confirmed_block(slot).await;
|
let bigtable_result = bigtable_ledger_storage.get_confirmed_block(slot).await;
|
||||||
|
|
Loading…
Reference in New Issue