| 
									
										
										
										
											2019-02-17 20:22:29 -07:00
										 |  |  | //! The `leader_confirmation_service` module implements the tools necessary
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  | //! to generate a thread which regularly calculates the last confirmation times
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | //! observed by the leader
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-07 20:16:27 -07:00
										 |  |  | use crate::service::Service;
 | 
					
						
							| 
									
										
										
										
											2018-11-16 08:45:59 -08:00
										 |  |  | use solana_metrics::{influxdb, submit};
 | 
					
						
							| 
									
										
										
										
											2019-02-18 23:26:22 -07:00
										 |  |  | use solana_runtime::bank::Bank;
 | 
					
						
							| 
									
										
										
										
											2018-12-08 16:54:42 -08:00
										 |  |  | use solana_sdk::pubkey::Pubkey;
 | 
					
						
							| 
									
										
										
										
											2018-11-16 08:45:59 -08:00
										 |  |  | use solana_sdk::timing;
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | use std::result;
 | 
					
						
							|  |  |  | use std::sync::atomic::{AtomicBool, Ordering};
 | 
					
						
							|  |  |  | use std::sync::Arc;
 | 
					
						
							|  |  |  | use std::thread::sleep;
 | 
					
						
							|  |  |  | use std::thread::{self, Builder, JoinHandle};
 | 
					
						
							|  |  |  | use std::time::Duration;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[derive(Debug, PartialEq, Eq)]
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  | pub enum ConfirmationError {
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |     NoValidSupermajority,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  | pub const COMPUTE_CONFIRMATION_MS: u64 = 100;
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-17 20:22:29 -07:00
										 |  |  | pub struct LeaderConfirmationService {
 | 
					
						
							|  |  |  |     thread_hdl: JoinHandle<()>,
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-17 20:22:29 -07:00
										 |  |  | impl LeaderConfirmationService {
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |     fn get_last_supermajority_timestamp(
 | 
					
						
							|  |  |  |         bank: &Arc<Bank>,
 | 
					
						
							| 
									
										
										
										
											2018-12-08 16:54:42 -08:00
										 |  |  |         leader_id: Pubkey,
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |         last_valid_validator_timestamp: u64,
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |     ) -> result::Result<u64, ConfirmationError> {
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |         let mut total_stake = 0;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-13 10:47:29 -08:00
										 |  |  |         // Hold an accounts_db read lock as briefly as possible, just long enough to collect all
 | 
					
						
							|  |  |  |         // the vote states
 | 
					
						
							| 
									
										
										
										
											2019-02-28 17:08:45 -08:00
										 |  |  |         let vote_states = bank.vote_states(|_, vote_state| leader_id != vote_state.delegate_id);
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-13 10:47:29 -08:00
										 |  |  |         let mut ticks_and_stakes: Vec<(u64, u64)> = vote_states
 | 
					
						
							|  |  |  |             .iter()
 | 
					
						
							| 
									
										
										
										
											2019-02-28 17:08:45 -08:00
										 |  |  |             .filter_map(|(_, vote_state)| {
 | 
					
						
							|  |  |  |                 let validator_stake = bank.get_balance(&vote_state.delegate_id);
 | 
					
						
							| 
									
										
										
										
											2019-01-13 10:47:29 -08:00
										 |  |  |                 total_stake += validator_stake;
 | 
					
						
							|  |  |  |                 // Filter out any validators that don't have at least one vote
 | 
					
						
							|  |  |  |                 // by returning None
 | 
					
						
							|  |  |  |                 vote_state
 | 
					
						
							|  |  |  |                     .votes
 | 
					
						
							|  |  |  |                     .back()
 | 
					
						
							| 
									
										
										
										
											2019-02-21 00:44:37 -08:00
										 |  |  |                     // A vote for a slot is like a vote for the last tick in that slot
 | 
					
						
							| 
									
										
										
										
											2019-02-21 01:43:57 -08:00
										 |  |  |                     .map(|vote| {
 | 
					
						
							|  |  |  |                         (
 | 
					
						
							|  |  |  |                             (vote.slot_height + 1) * bank.ticks_per_slot() - 1,
 | 
					
						
							|  |  |  |                             validator_stake,
 | 
					
						
							|  |  |  |                         )
 | 
					
						
							|  |  |  |                     })
 | 
					
						
							| 
									
										
										
										
											2019-01-13 10:47:29 -08:00
										 |  |  |             })
 | 
					
						
							|  |  |  |             .collect();
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         let super_majority_stake = (2 * total_stake) / 3;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if let Some(last_valid_validator_timestamp) =
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |             bank.get_confirmation_timestamp(&mut ticks_and_stakes, super_majority_stake)
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |         {
 | 
					
						
							|  |  |  |             return Ok(last_valid_validator_timestamp);
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if last_valid_validator_timestamp != 0 {
 | 
					
						
							| 
									
										
										
										
											2019-02-08 13:23:28 -08:00
										 |  |  |             let now = timing::timestamp();
 | 
					
						
							| 
									
										
										
										
											2018-11-16 08:45:59 -08:00
										 |  |  |             submit(
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |                 influxdb::Point::new(&"leader-confirmation")
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |                     .add_field(
 | 
					
						
							|  |  |  |                         "duration_ms",
 | 
					
						
							|  |  |  |                         influxdb::Value::Integer((now - last_valid_validator_timestamp) as i64),
 | 
					
						
							| 
									
										
										
										
											2018-12-07 20:01:28 -07:00
										 |  |  |                     )
 | 
					
						
							|  |  |  |                     .to_owned(),
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |             );
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |         Err(ConfirmationError::NoValidSupermajority)
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |     pub fn compute_confirmation(
 | 
					
						
							| 
									
										
										
										
											2018-12-08 16:54:42 -08:00
										 |  |  |         bank: &Arc<Bank>,
 | 
					
						
							|  |  |  |         leader_id: Pubkey,
 | 
					
						
							|  |  |  |         last_valid_validator_timestamp: &mut u64,
 | 
					
						
							|  |  |  |     ) {
 | 
					
						
							| 
									
										
										
										
											2019-02-21 01:43:57 -08:00
										 |  |  |         if let Ok(super_majority_timestamp) =
 | 
					
						
							|  |  |  |             Self::get_last_supermajority_timestamp(bank, leader_id, *last_valid_validator_timestamp)
 | 
					
						
							|  |  |  |         {
 | 
					
						
							| 
									
										
										
										
											2019-02-08 13:23:28 -08:00
										 |  |  |             let now = timing::timestamp();
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |             let confirmation_ms = now - super_majority_timestamp;
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |             *last_valid_validator_timestamp = super_majority_timestamp;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 08:45:59 -08:00
										 |  |  |             submit(
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |                 influxdb::Point::new(&"leader-confirmation")
 | 
					
						
							|  |  |  |                     .add_field(
 | 
					
						
							|  |  |  |                         "duration_ms",
 | 
					
						
							|  |  |  |                         influxdb::Value::Integer(confirmation_ms as i64),
 | 
					
						
							|  |  |  |                     )
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |                     .to_owned(),
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-17 20:22:29 -07:00
										 |  |  |     /// Create a new LeaderConfirmationService for computing confirmation.
 | 
					
						
							| 
									
										
										
										
											2019-02-27 10:23:27 -08:00
										 |  |  |     pub fn new(bank: &Arc<Bank>, leader_id: Pubkey, exit: Arc<AtomicBool>) -> Self {
 | 
					
						
							|  |  |  |         let bank = bank.clone();
 | 
					
						
							| 
									
										
										
										
											2019-02-17 20:22:29 -07:00
										 |  |  |         let thread_hdl = Builder::new()
 | 
					
						
							| 
									
										
										
										
											2019-02-21 00:44:37 -08:00
										 |  |  |             .name("solana-leader-confirmation-service".to_string())
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |             .spawn(move || {
 | 
					
						
							|  |  |  |                 let mut last_valid_validator_timestamp = 0;
 | 
					
						
							|  |  |  |                 loop {
 | 
					
						
							|  |  |  |                     if exit.load(Ordering::Relaxed) {
 | 
					
						
							|  |  |  |                         break;
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |                     Self::compute_confirmation(
 | 
					
						
							|  |  |  |                         &bank,
 | 
					
						
							|  |  |  |                         leader_id,
 | 
					
						
							|  |  |  |                         &mut last_valid_validator_timestamp,
 | 
					
						
							|  |  |  |                     );
 | 
					
						
							|  |  |  |                     sleep(Duration::from_millis(COMPUTE_CONFIRMATION_MS));
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |                 }
 | 
					
						
							| 
									
										
										
										
											2018-12-07 20:01:28 -07:00
										 |  |  |             })
 | 
					
						
							|  |  |  |             .unwrap();
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-17 20:22:29 -07:00
										 |  |  |         Self { thread_hdl }
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-17 20:22:29 -07:00
										 |  |  | impl Service for LeaderConfirmationService {
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |     type JoinReturnType = ();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn join(self) -> thread::Result<()> {
 | 
					
						
							| 
									
										
										
										
											2019-02-17 20:22:29 -07:00
										 |  |  |         self.thread_hdl.join()
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[cfg(test)]
 | 
					
						
							| 
									
										
										
										
											2019-02-23 22:00:55 -07:00
										 |  |  | mod tests {
 | 
					
						
							| 
									
										
										
										
											2019-02-17 20:22:29 -07:00
										 |  |  |     use super::*;
 | 
					
						
							| 
									
										
										
										
											2019-02-23 22:00:55 -07:00
										 |  |  |     use crate::voting_keypair::tests::{new_vote_account, push_vote};
 | 
					
						
							| 
									
										
										
										
											2019-02-18 23:26:22 -07:00
										 |  |  |     use crate::voting_keypair::VotingKeypair;
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |     use bincode::serialize;
 | 
					
						
							| 
									
										
										
										
											2019-02-18 23:26:22 -07:00
										 |  |  |     use solana_sdk::genesis_block::GenesisBlock;
 | 
					
						
							| 
									
										
										
										
											2018-11-16 08:04:46 -08:00
										 |  |  |     use solana_sdk::hash::hash;
 | 
					
						
							| 
									
										
										
										
											2018-12-03 10:26:28 -08:00
										 |  |  |     use solana_sdk::signature::{Keypair, KeypairUtil};
 | 
					
						
							| 
									
										
										
										
											2019-01-29 17:16:59 -07:00
										 |  |  |     use solana_sdk::vote_transaction::VoteTransaction;
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |     use std::sync::Arc;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[test]
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |     fn test_compute_confirmation() {
 | 
					
						
							| 
									
										
										
										
											2018-12-14 12:36:50 -08:00
										 |  |  |         solana_logger::setup();
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-24 12:04:04 -08:00
										 |  |  |         let (genesis_block, mint_keypair) = GenesisBlock::new(1234);
 | 
					
						
							|  |  |  |         let bank = Arc::new(Bank::new(&genesis_block));
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |         // generate 10 validators, but only vote for the first 6 validators
 | 
					
						
							| 
									
										
										
										
											2019-02-21 01:43:57 -08:00
										 |  |  |         let ids: Vec<_> = (0..10 * bank.ticks_per_slot())
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |             .map(|i| {
 | 
					
						
							|  |  |  |                 let last_id = hash(&serialize(&i).unwrap()); // Unique hash
 | 
					
						
							| 
									
										
										
										
											2018-11-05 09:47:41 -08:00
										 |  |  |                 bank.register_tick(&last_id);
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |                 last_id
 | 
					
						
							| 
									
										
										
										
											2018-12-07 20:01:28 -07:00
										 |  |  |             })
 | 
					
						
							|  |  |  |             .collect();
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Create a total of 10 vote accounts, each will have a balance of 1 (after giving 1 to
 | 
					
						
							|  |  |  |         // their vote account), for a total staking pool of 10 tokens.
 | 
					
						
							|  |  |  |         let vote_accounts: Vec<_> = (0..10)
 | 
					
						
							|  |  |  |             .map(|i| {
 | 
					
						
							|  |  |  |                 // Create new validator to vote
 | 
					
						
							| 
									
										
										
										
											2019-01-10 09:21:38 -08:00
										 |  |  |                 let validator_keypair = Arc::new(Keypair::new());
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |                 let last_id = ids[i];
 | 
					
						
							| 
									
										
										
										
											2019-01-31 21:12:51 -07:00
										 |  |  |                 let voting_keypair = VotingKeypair::new_local(&validator_keypair);
 | 
					
						
							| 
									
										
										
										
											2019-02-20 10:29:29 -07:00
										 |  |  |                 let voting_pubkey = voting_keypair.pubkey();
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 // Give the validator some tokens
 | 
					
						
							| 
									
										
										
										
											2019-01-24 12:04:04 -08:00
										 |  |  |                 bank.transfer(2, &mint_keypair, validator_keypair.pubkey(), last_id)
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |                     .unwrap();
 | 
					
						
							| 
									
										
										
										
											2019-02-20 10:29:29 -07:00
										 |  |  |                 new_vote_account(&validator_keypair, &voting_pubkey, &bank, 1);
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if i < 6 {
 | 
					
						
							| 
									
										
										
										
											2019-02-20 10:29:29 -07:00
										 |  |  |                     push_vote(&voting_keypair, &bank, (i + 1) as u64);
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |                 }
 | 
					
						
							| 
									
										
										
										
											2019-01-31 21:12:51 -07:00
										 |  |  |                 (voting_keypair, validator_keypair)
 | 
					
						
							| 
									
										
										
										
											2018-12-07 20:01:28 -07:00
										 |  |  |             })
 | 
					
						
							|  |  |  |             .collect();
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |         // There isn't 2/3 consensus, so the bank's confirmation value should be the default
 | 
					
						
							|  |  |  |         let mut last_confirmation_time = 0;
 | 
					
						
							| 
									
										
										
										
											2019-02-17 20:22:29 -07:00
										 |  |  |         LeaderConfirmationService::compute_confirmation(
 | 
					
						
							| 
									
										
										
										
											2018-12-08 16:54:42 -08:00
										 |  |  |             &bank,
 | 
					
						
							| 
									
										
										
										
											2019-02-05 08:03:52 -08:00
										 |  |  |             genesis_block.bootstrap_leader_id,
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |             &mut last_confirmation_time,
 | 
					
						
							| 
									
										
										
										
											2018-12-08 16:54:42 -08:00
										 |  |  |         );
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Get another validator to vote, so we now have 2/3 consensus
 | 
					
						
							| 
									
										
										
										
											2019-01-31 21:12:51 -07:00
										 |  |  |         let voting_keypair = &vote_accounts[7].0;
 | 
					
						
							| 
									
										
										
										
											2019-02-01 08:36:35 -07:00
										 |  |  |         let vote_tx = VoteTransaction::new_vote(voting_keypair, 7, ids[6], 0);
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |         bank.process_transaction(&vote_tx).unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-17 20:22:29 -07:00
										 |  |  |         LeaderConfirmationService::compute_confirmation(
 | 
					
						
							| 
									
										
										
										
											2018-12-08 16:54:42 -08:00
										 |  |  |             &bank,
 | 
					
						
							| 
									
										
										
										
											2019-02-05 08:03:52 -08:00
										 |  |  |             genesis_block.bootstrap_leader_id,
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |             &mut last_confirmation_time,
 | 
					
						
							| 
									
										
										
										
											2018-12-08 16:54:42 -08:00
										 |  |  |         );
 | 
					
						
							| 
									
										
										
										
											2018-12-20 15:47:48 -08:00
										 |  |  |         assert!(last_confirmation_time > 0);
 | 
					
						
							| 
									
										
										
										
											2018-11-02 15:49:14 -07:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 |