use {
crate::stakes::{DeserializableStakes, SerdeStakesToStakeFormat, Stakes},
serde::{Deserialize, Serialize},
solana_bls_signatures::{Pubkey as BLSPubkey, PubkeyCompressed as BLSPubkeyCompressed},
solana_clock::Epoch,
solana_pubkey::Pubkey,
solana_stake_interface::state::Stake,
solana_vote::vote_account::VoteAccountsHashMap,
std::{
collections::HashMap,
sync::{Arc, OnceLock},
},
};
pub type NodeIdToVoteAccounts = HashMap<Pubkey, NodeVoteAccounts>;
pub type EpochAuthorizedVoters = HashMap<Pubkey, Pubkey>;
#[derive(Clone, Debug, Default)]
#[cfg_attr(feature = "frozen-abi", derive(AbiExample))]
#[cfg_attr(feature = "dev-context-only-utils", derive(PartialEq))]
pub struct BLSPubkeyStakeEntry {
pub pubkey: Pubkey,
pub bls_pubkey: BLSPubkey,
pub stake: u64,
}
#[derive(Clone, Debug, Default)]
#[cfg_attr(feature = "frozen-abi", derive(AbiExample))]
#[cfg_attr(feature = "dev-context-only-utils", derive(PartialEq))]
pub struct BLSPubkeyToRankMap {
rank_map: HashMap<BLSPubkey, u16>,
sorted_pubkeys: Vec<BLSPubkeyStakeEntry>,
}
impl BLSPubkeyToRankMap {
pub fn new(epoch_vote_accounts_hash_map: &VoteAccountsHashMap) -> Self {
let mut pubkey_stake_pair_vec: Vec<(Pubkey, BLSPubkey, u64)> = epoch_vote_accounts_hash_map
.iter()
.filter_map(|(pubkey, (stake, account))| {
if *stake > 0 {
account
.vote_state_view()
.bls_pubkey_compressed()
.and_then(|bls_pubkey_compressed_bytes| {
let bls_pubkey_compressed =
BLSPubkeyCompressed(bls_pubkey_compressed_bytes);
BLSPubkey::try_from(bls_pubkey_compressed).ok()
})
.map(|bls_pubkey| (*pubkey, bls_pubkey, *stake))
} else {
None
}
})
.collect();
pubkey_stake_pair_vec.sort_by(|(_, a_pubkey, a_stake), (_, b_pubkey, b_stake)| {
b_stake.cmp(a_stake).then(a_pubkey.cmp(b_pubkey))
});
let mut sorted_pubkeys = Vec::new();
let mut bls_pubkey_to_rank_map = HashMap::new();
for (rank, (pubkey, bls_pubkey, stake)) in pubkey_stake_pair_vec.into_iter().enumerate() {
let entry = BLSPubkeyStakeEntry {
pubkey,
bls_pubkey,
stake,
};
sorted_pubkeys.push(entry);
bls_pubkey_to_rank_map.insert(bls_pubkey, rank as u16);
}
Self {
rank_map: bls_pubkey_to_rank_map,
sorted_pubkeys,
}
}
pub fn is_empty(&self) -> bool {
self.rank_map.is_empty()
}
pub fn len(&self) -> usize {
self.rank_map.len()
}
pub fn get_rank(&self, bls_pubkey: &BLSPubkey) -> Option<&u16> {
self.rank_map.get(bls_pubkey)
}
pub fn get_pubkey_stake_entry(&self, index: usize) -> Option<&BLSPubkeyStakeEntry> {
self.sorted_pubkeys.get(index)
}
}
#[cfg_attr(feature = "frozen-abi", derive(AbiExample))]
#[derive(Clone, Serialize, Debug, Deserialize, Default, PartialEq, Eq)]
pub struct NodeVoteAccounts {
pub vote_accounts: Vec<Pubkey>,
pub total_stake: u64,
}
#[derive(Clone, Debug, Deserialize)]
pub(crate) enum DeserializableVersionedEpochStakes {
Current {
stakes: DeserializableStakes<Stake>,
total_stake: u64,
node_id_to_vote_accounts: NodeIdToVoteAccounts,
epoch_authorized_voters: EpochAuthorizedVoters,
},
}
#[derive(Clone, Debug, Serialize)]
#[cfg_attr(feature = "frozen-abi", derive(AbiExample, AbiEnumVisitor))]
#[cfg_attr(feature = "dev-context-only-utils", derive(PartialEq))]
pub enum VersionedEpochStakes {
Current {
stakes: SerdeStakesToStakeFormat,
total_stake: u64,
node_id_to_vote_accounts: Arc<NodeIdToVoteAccounts>,
epoch_authorized_voters: Arc<EpochAuthorizedVoters>,
#[serde(skip)]
bls_pubkey_to_rank_map: OnceLock<Arc<BLSPubkeyToRankMap>>,
},
}
impl From<DeserializableVersionedEpochStakes> for VersionedEpochStakes {
fn from(epoch_stakes: DeserializableVersionedEpochStakes) -> Self {
let DeserializableVersionedEpochStakes::Current {
stakes,
total_stake,
node_id_to_vote_accounts,
epoch_authorized_voters,
} = epoch_stakes;
Self::Current {
stakes: SerdeStakesToStakeFormat::Stake(Stakes::from_deserialized(stakes)),
total_stake,
node_id_to_vote_accounts: Arc::new(node_id_to_vote_accounts),
epoch_authorized_voters: Arc::new(epoch_authorized_voters),
bls_pubkey_to_rank_map: OnceLock::new(),
}
}
}
impl VersionedEpochStakes {
pub(crate) fn new(stakes: SerdeStakesToStakeFormat, leader_schedule_epoch: Epoch) -> Self {
let epoch_vote_accounts = stakes.vote_accounts();
let (total_stake, node_id_to_vote_accounts, epoch_authorized_voters) =
Self::parse_epoch_vote_accounts(epoch_vote_accounts.as_ref(), leader_schedule_epoch);
Self::Current {
stakes,
total_stake,
node_id_to_vote_accounts: Arc::new(node_id_to_vote_accounts),
epoch_authorized_voters: Arc::new(epoch_authorized_voters),
bls_pubkey_to_rank_map: OnceLock::new(),
}
}
#[cfg(feature = "dev-context-only-utils")]
pub fn new_for_tests(
vote_accounts_hash_map: VoteAccountsHashMap,
leader_schedule_epoch: Epoch,
) -> Self {
Self::new(
SerdeStakesToStakeFormat::Account(crate::stakes::Stakes::new_for_tests(
0,
solana_vote::vote_account::VoteAccounts::from(Arc::new(vote_accounts_hash_map)),
im::HashMap::default(),
)),
leader_schedule_epoch,
)
}
pub fn stakes(&self) -> &SerdeStakesToStakeFormat {
match self {
Self::Current { stakes, .. } => stakes,
}
}
pub fn total_stake(&self) -> u64 {
match self {
Self::Current { total_stake, .. } => *total_stake,
}
}
#[cfg(feature = "dev-context-only-utils")]
pub fn set_total_stake(&mut self, total_stake: u64) {
match self {
Self::Current {
total_stake: total_stake_field,
..
} => {
*total_stake_field = total_stake;
}
}
}
pub fn node_id_to_vote_accounts(&self) -> &Arc<NodeIdToVoteAccounts> {
match self {
Self::Current {
node_id_to_vote_accounts,
..
} => node_id_to_vote_accounts,
}
}
pub fn node_id_to_stake(&self, node_id: &Pubkey) -> Option<u64> {
self.node_id_to_vote_accounts()
.get(node_id)
.map(|x| x.total_stake)
}
pub fn epoch_authorized_voters(&self) -> &Arc<EpochAuthorizedVoters> {
match self {
Self::Current {
epoch_authorized_voters,
..
} => epoch_authorized_voters,
}
}
pub fn bls_pubkey_to_rank_map(&self) -> &Arc<BLSPubkeyToRankMap> {
match self {
Self::Current {
bls_pubkey_to_rank_map,
..
} => bls_pubkey_to_rank_map.get_or_init(|| {
Arc::new(BLSPubkeyToRankMap::new(
self.stakes().vote_accounts().as_ref(),
))
}),
}
}
pub fn vote_account_stake(&self, vote_account: &Pubkey) -> u64 {
self.stakes()
.vote_accounts()
.get_delegated_stake(vote_account)
}
fn parse_epoch_vote_accounts(
epoch_vote_accounts: &VoteAccountsHashMap,
leader_schedule_epoch: Epoch,
) -> (u64, NodeIdToVoteAccounts, EpochAuthorizedVoters) {
let mut node_id_to_vote_accounts: NodeIdToVoteAccounts = HashMap::new();
let mut epoch_authorized_voters: EpochAuthorizedVoters = HashMap::new();
let mut total_stake: u64 = 0;
for (key, (stake, account)) in epoch_vote_accounts.iter() {
total_stake += *stake;
if *stake == 0 {
continue;
}
let vote_state = account.vote_state_view();
if let Some(authorized_voter) = vote_state.get_authorized_voter(leader_schedule_epoch) {
let node_vote_accounts = node_id_to_vote_accounts
.entry(*vote_state.node_pubkey())
.or_default();
node_vote_accounts.total_stake += stake;
node_vote_accounts.vote_accounts.push(*key);
epoch_authorized_voters.insert(*key, *authorized_voter);
}
}
(
total_stake,
node_id_to_vote_accounts,
epoch_authorized_voters,
)
}
}
#[cfg(test)]
pub(crate) mod tests {
use {
super::*, solana_account::AccountSharedData,
solana_bls_signatures::keypair::Keypair as BLSKeypair,
solana_vote::vote_account::VoteAccount,
solana_vote_interface::state::BLS_PUBLIC_KEY_COMPRESSED_SIZE,
solana_vote_program::vote_state::create_v4_account_with_authorized, std::iter,
test_case::test_case,
};
struct VoteAccountInfo {
vote_account: Pubkey,
account: AccountSharedData,
authorized_voter: Pubkey,
}
fn new_vote_accounts(
num_nodes: usize,
num_vote_accounts_per_node: usize,
is_alpenglow: bool,
) -> HashMap<Pubkey, Vec<VoteAccountInfo>> {
(0..num_nodes)
.map(|_| {
let node_id = solana_pubkey::new_rand();
(
node_id,
iter::repeat_with(|| {
let authorized_voter = solana_pubkey::new_rand();
let bls_pubkey_compressed: BLSPubkeyCompressed =
BLSKeypair::new().public.into();
let bls_pubkey_compressed_serialized =
bincode::serialize(&bls_pubkey_compressed)
.unwrap()
.try_into()
.unwrap();
let bls_pubkey = if is_alpenglow {
bls_pubkey_compressed_serialized
} else {
[0u8; BLS_PUBLIC_KEY_COMPRESSED_SIZE]
};
let account = create_v4_account_with_authorized(
&node_id,
&authorized_voter,
bls_pubkey,
&node_id,
0,
&node_id,
0,
&node_id,
100,
);
VoteAccountInfo {
vote_account: solana_pubkey::new_rand(),
account,
authorized_voter,
}
})
.take(num_vote_accounts_per_node)
.collect(),
)
})
.collect()
}
fn new_epoch_vote_accounts(
vote_accounts_map: &HashMap<Pubkey, Vec<VoteAccountInfo>>,
node_id_to_stake_fn: impl Fn(&Pubkey) -> u64,
) -> VoteAccountsHashMap {
vote_accounts_map
.iter()
.flat_map(|(node_id, vote_accounts)| {
vote_accounts.iter().map(|v| {
let vote_account = VoteAccount::try_from(v.account.clone()).unwrap();
(v.vote_account, (node_id_to_stake_fn(node_id), vote_account))
})
})
.collect()
}
#[test_case(true; "alpenglow")]
#[test_case(false; "towerbft")]
fn test_parse_epoch_vote_accounts(is_alpenglow: bool) {
let stake_per_account = 100;
let num_vote_accounts_per_node = 2;
let num_nodes = 10;
let vote_accounts_map =
new_vote_accounts(num_nodes, num_vote_accounts_per_node, is_alpenglow);
let expected_authorized_voters: HashMap<_, _> = vote_accounts_map
.iter()
.flat_map(|(_, vote_accounts)| {
vote_accounts
.iter()
.map(|v| (v.vote_account, v.authorized_voter))
})
.collect();
let expected_node_id_to_vote_accounts: HashMap<_, _> = vote_accounts_map
.iter()
.map(|(node_pubkey, vote_accounts)| {
let mut vote_accounts = vote_accounts
.iter()
.map(|v| v.vote_account)
.collect::<Vec<_>>();
vote_accounts.sort();
let node_vote_accounts = NodeVoteAccounts {
vote_accounts,
total_stake: stake_per_account * num_vote_accounts_per_node as u64,
};
(*node_pubkey, node_vote_accounts)
})
.collect();
let epoch_vote_accounts =
new_epoch_vote_accounts(&vote_accounts_map, |_| stake_per_account);
let (total_stake, mut node_id_to_vote_accounts, epoch_authorized_voters) =
VersionedEpochStakes::parse_epoch_vote_accounts(&epoch_vote_accounts, 0);
node_id_to_vote_accounts
.iter_mut()
.for_each(|(_, node_vote_accounts)| node_vote_accounts.vote_accounts.sort());
assert!(
node_id_to_vote_accounts.len() == expected_node_id_to_vote_accounts.len()
&& node_id_to_vote_accounts
.iter()
.all(|(k, v)| expected_node_id_to_vote_accounts.get(k).unwrap() == v)
);
assert!(
epoch_authorized_voters.len() == expected_authorized_voters.len()
&& epoch_authorized_voters
.iter()
.all(|(k, v)| expected_authorized_voters.get(k).unwrap() == v)
);
assert_eq!(
total_stake,
num_nodes as u64 * num_vote_accounts_per_node as u64 * 100
);
}
#[test_case(true; "alpenglow")]
#[test_case(false; "towerbft")]
fn test_node_id_to_stake(is_alpenglow: bool) {
let num_nodes = 10;
let num_vote_accounts_per_node = 2;
let vote_accounts_map =
new_vote_accounts(num_nodes, num_vote_accounts_per_node, is_alpenglow);
let node_id_to_stake_map = vote_accounts_map
.keys()
.enumerate()
.map(|(index, node_id)| (*node_id, ((index + 1) * 100) as u64))
.collect::<HashMap<_, _>>();
let epoch_vote_accounts = new_epoch_vote_accounts(&vote_accounts_map, |node_id| {
*node_id_to_stake_map.get(node_id).unwrap()
});
let epoch_stakes = VersionedEpochStakes::new_for_tests(epoch_vote_accounts, 0);
assert_eq!(epoch_stakes.total_stake(), 11000);
for (node_id, stake) in node_id_to_stake_map.iter() {
assert_eq!(
epoch_stakes.node_id_to_stake(node_id),
Some(*stake * num_vote_accounts_per_node as u64)
);
}
}
#[test_case(1; "single_vote_account")]
#[test_case(2; "multiple_vote_accounts")]
fn test_bls_pubkey_rank_map(num_vote_accounts_per_node: usize) {
agave_logger::setup();
let num_nodes = 10;
let num_vote_accounts = num_nodes * num_vote_accounts_per_node;
let vote_accounts_map = new_vote_accounts(num_nodes, num_vote_accounts_per_node, true);
let node_id_to_stake_map = vote_accounts_map
.keys()
.enumerate()
.map(|(index, node_id)| (*node_id, ((index + 1) * 100) as u64))
.collect::<HashMap<_, _>>();
let epoch_vote_accounts = new_epoch_vote_accounts(&vote_accounts_map, |node_id| {
*node_id_to_stake_map.get(node_id).unwrap()
});
let epoch_stakes = VersionedEpochStakes::new_for_tests(epoch_vote_accounts.clone(), 0);
let bls_pubkey_to_rank_map = epoch_stakes.bls_pubkey_to_rank_map();
assert_eq!(bls_pubkey_to_rank_map.len(), num_vote_accounts);
for (pubkey, (stake, vote_account)) in epoch_vote_accounts {
let vote_state_view = vote_account.vote_state_view();
let bls_pubkey_compressed = bincode::deserialize::<BLSPubkeyCompressed>(
&vote_state_view.bls_pubkey_compressed().unwrap(),
)
.unwrap();
let bls_pubkey = BLSPubkey::try_from(bls_pubkey_compressed).unwrap();
let index = bls_pubkey_to_rank_map.get_rank(&bls_pubkey).unwrap();
assert!(index >= &0 && index < &(num_vote_accounts as u16));
assert_eq!(
bls_pubkey_to_rank_map.get_pubkey_stake_entry(*index as usize),
Some(&BLSPubkeyStakeEntry {
pubkey,
bls_pubkey,
stake,
})
);
}
let mut bank_epoch_stakes = HashMap::new();
bank_epoch_stakes.insert(0, epoch_stakes.clone());
let epoch_stakes = bank_epoch_stakes
.get(&0)
.expect("Epoch stakes should exist");
let bls_pubkey_to_rank_map2 = epoch_stakes.bls_pubkey_to_rank_map();
assert_eq!(bls_pubkey_to_rank_map2, bls_pubkey_to_rank_map);
}
}