use {
crate::optimistically_confirmed_bank_tracker::OptimisticallyConfirmedBank,
solana_clock::Slot,
solana_ledger::blockstore::Blockstore,
std::sync::{
atomic::{AtomicBool, Ordering},
Arc, RwLock,
},
};
#[derive(PartialEq, Eq, Clone, Copy, Debug)]
pub enum RpcHealthStatus {
Ok,
Behind { num_slots: Slot }, Unknown,
}
pub struct RpcHealth {
optimistically_confirmed_bank: Arc<RwLock<OptimisticallyConfirmedBank>>,
blockstore: Arc<Blockstore>,
health_check_slot_distance: u64,
override_health_check: Arc<AtomicBool>,
#[cfg(test)]
stub_health_status: std::sync::RwLock<Option<RpcHealthStatus>>,
}
impl RpcHealth {
pub fn new(
optimistically_confirmed_bank: Arc<RwLock<OptimisticallyConfirmedBank>>,
blockstore: Arc<Blockstore>,
health_check_slot_distance: u64,
override_health_check: Arc<AtomicBool>,
) -> Self {
Self {
optimistically_confirmed_bank,
blockstore,
health_check_slot_distance,
override_health_check,
#[cfg(test)]
stub_health_status: std::sync::RwLock::new(None),
}
}
pub fn check(&self) -> RpcHealthStatus {
#[cfg(test)]
{
if let Some(stub_health_status) = *self.stub_health_status.read().unwrap() {
return stub_health_status;
}
}
if self.override_health_check.load(Ordering::Relaxed) {
return RpcHealthStatus::Ok;
}
let my_latest_optimistically_confirmed_slot = self
.optimistically_confirmed_bank
.read()
.unwrap()
.bank
.slot();
let mut optimistic_slot_infos = match self.blockstore.get_latest_optimistic_slots(1) {
Ok(infos) => infos,
Err(err) => {
warn!("health check: blockstore error: {err}");
return RpcHealthStatus::Unknown;
}
};
let Some((cluster_latest_optimistically_confirmed_slot, _, _)) =
optimistic_slot_infos.pop()
else {
warn!("health check: blockstore does not contain any optimistically confirmed slots");
return RpcHealthStatus::Unknown;
};
if my_latest_optimistically_confirmed_slot
>= cluster_latest_optimistically_confirmed_slot
.saturating_sub(self.health_check_slot_distance)
{
RpcHealthStatus::Ok
} else {
let num_slots = cluster_latest_optimistically_confirmed_slot
.saturating_sub(my_latest_optimistically_confirmed_slot);
warn!(
"health check: behind by {num_slots} slots: \
me={my_latest_optimistically_confirmed_slot}, latest \
cluster={cluster_latest_optimistically_confirmed_slot}",
);
RpcHealthStatus::Behind { num_slots }
}
}
#[cfg(test)]
pub(crate) fn stub(
optimistically_confirmed_bank: Arc<RwLock<OptimisticallyConfirmedBank>>,
blockstore: Arc<Blockstore>,
) -> Arc<Self> {
Arc::new(Self::new(
optimistically_confirmed_bank,
blockstore,
42,
Arc::new(AtomicBool::new(false)),
))
}
#[cfg(test)]
pub(crate) fn stub_set_health_status(&self, stub_health_status: Option<RpcHealthStatus>) {
*self.stub_health_status.write().unwrap() = stub_health_status;
}
}
#[cfg(test)]
pub mod tests {
use {
super::*,
solana_clock::UnixTimestamp,
solana_hash::Hash,
solana_ledger::{
genesis_utils::{create_genesis_config, GenesisConfigInfo},
get_tmp_ledger_path_auto_delete,
},
solana_pubkey::Pubkey,
solana_runtime::{bank::Bank, bank_forks::BankForks},
};
#[test]
fn test_get_health() {
let ledger_path = get_tmp_ledger_path_auto_delete!();
let blockstore = Arc::new(Blockstore::open(ledger_path.path()).unwrap());
let GenesisConfigInfo { genesis_config, .. } = create_genesis_config(100);
let bank = Bank::new_for_tests(&genesis_config);
let bank_forks = BankForks::new_rw_arc(bank);
let optimistically_confirmed_bank =
OptimisticallyConfirmedBank::locked_from_bank_forks_root(&bank_forks);
let bank0 = bank_forks.read().unwrap().root_bank();
assert!(bank0.slot() == 0);
let health_check_slot_distance = 10;
let override_health_check = Arc::new(AtomicBool::new(true));
let health = RpcHealth::new(
optimistically_confirmed_bank.clone(),
blockstore.clone(),
health_check_slot_distance,
override_health_check.clone(),
);
assert_eq!(health.check(), RpcHealthStatus::Ok);
override_health_check.store(false, Ordering::Relaxed);
assert_eq!(health.check(), RpcHealthStatus::Unknown);
blockstore
.insert_optimistic_slot(15, &Hash::default(), UnixTimestamp::default())
.unwrap();
assert_eq!(health.check(), RpcHealthStatus::Behind { num_slots: 15 });
let bank4 = Arc::new(Bank::new_from_parent(bank0, &Pubkey::default(), 4));
optimistically_confirmed_bank.write().unwrap().bank = bank4.clone();
assert_eq!(health.check(), RpcHealthStatus::Behind { num_slots: 11 });
let bank5 = Arc::new(Bank::new_from_parent(bank4, &Pubkey::default(), 5));
optimistically_confirmed_bank.write().unwrap().bank = bank5.clone();
assert_eq!(health.check(), RpcHealthStatus::Ok);
let bank15 = Arc::new(Bank::new_from_parent(bank5, &Pubkey::default(), 15));
optimistically_confirmed_bank.write().unwrap().bank = bank15.clone();
assert_eq!(health.check(), RpcHealthStatus::Ok);
let bank16 = Arc::new(Bank::new_from_parent(bank15, &Pubkey::default(), 16));
optimistically_confirmed_bank.write().unwrap().bank = bank16.clone();
assert_eq!(health.check(), RpcHealthStatus::Ok);
}
}