#[macro_use] extern crate log; #[cfg(feature = "chacha")] #[macro_use] extern crate serde_json; use bincode::deserialize; use solana::blocktree::{ create_tmp_sample_ledger, get_tmp_ledger_path, tmp_copy_ledger, Blocktree, DEFAULT_SLOT_HEIGHT, }; use solana::client::mk_client; use solana::cluster_info::{ClusterInfo, Node, NodeInfo}; use solana::entry::Entry; use solana::fullnode::{Fullnode, FullnodeConfig}; use solana::replicator::Replicator; use solana::storage_stage::STORAGE_ROTATE_TEST_COUNT; use solana::streamer::blob_receiver; use solana::voting_keypair::VotingKeypair; use solana_sdk::hash::Hash; use solana_sdk::signature::{Keypair, KeypairUtil}; use solana_sdk::system_transaction::SystemTransaction; use std::fs::remove_dir_all; use std::sync::atomic::{AtomicBool, Ordering}; use std::sync::mpsc::channel; use std::sync::Arc; use std::time::Duration; #[test] #[ignore] fn test_replicator_startup_basic() { solana_logger::setup(); info!("starting replicator test"); let replicator_ledger_path = &get_tmp_ledger_path("replicator_test_replicator_ledger"); info!("starting leader node"); let leader_keypair = Arc::new(Keypair::new()); let leader_node = Node::new_localhost_with_pubkey(leader_keypair.pubkey()); let leader_info = leader_node.info.clone(); let leader_ledger_path = "replicator_test_leader_ledger"; let mut fullnode_config = FullnodeConfig::default(); let blocktree_config = fullnode_config.ledger_config(); let ( mint_keypair, leader_ledger_path, _tick_height, _last_entry_height, _last_id, _last_entry_id, ) = create_tmp_sample_ledger( leader_ledger_path, 1_000_000_000, 0, leader_info.id, 42, &blocktree_config, ); let validator_ledger_path = tmp_copy_ledger( &leader_ledger_path, "replicator_test_validator_ledger", &blocktree_config, ); { let voting_keypair = VotingKeypair::new_local(&leader_keypair); fullnode_config.storage_rotate_count = STORAGE_ROTATE_TEST_COUNT; let leader = Fullnode::new( leader_node, &leader_keypair, &leader_ledger_path, voting_keypair, None, &fullnode_config, ); let leader_exit = leader.run(None); debug!( "leader: {:?}", solana::thin_client::poll_gossip_for_leader(leader_info.gossip, Some(5)).unwrap() ); let validator_keypair = Arc::new(Keypair::new()); let voting_keypair = VotingKeypair::new_local(&validator_keypair); let mut leader_client = mk_client(&leader_info); let last_id = leader_client.get_last_id(); debug!("last_id: {:?}", last_id); leader_client .transfer(10, &mint_keypair, validator_keypair.pubkey(), &last_id) .unwrap(); let validator_node = Node::new_localhost_with_pubkey(validator_keypair.pubkey()); #[cfg(feature = "chacha")] let validator_node_info = validator_node.info.clone(); let validator = Fullnode::new( validator_node, &validator_keypair, &validator_ledger_path, voting_keypair, Some(&leader_info), &fullnode_config, ); let validator_exit = validator.run(None); let bob = Keypair::new(); info!("starting transfers.."); for i in 0..64 { debug!("transfer {}", i); let last_id = leader_client.get_last_id(); let mut transaction = SystemTransaction::new_account(&mint_keypair, bob.pubkey(), 1, last_id, 0); leader_client .retry_transfer(&mint_keypair, &mut transaction, 5) .unwrap(); debug!( "transfer {}: mint balance={:?}, bob balance={:?}", i, leader_client.get_balance(&mint_keypair.pubkey()), leader_client.get_balance(&bob.pubkey()), ); } let replicator_keypair = Keypair::new(); info!("giving replicator tokens.."); let last_id = leader_client.get_last_id(); // Give the replicator some tokens let mut tx = SystemTransaction::new_account( &mint_keypair, replicator_keypair.pubkey(), 1, last_id, 0, ); leader_client .retry_transfer(&mint_keypair, &mut tx, 5) .unwrap(); info!("starting replicator node"); let replicator_node = Node::new_localhost_with_pubkey(replicator_keypair.pubkey()); let replicator_info = replicator_node.info.clone(); let leader_info = NodeInfo::new_entry_point(&leader_info.gossip); let replicator = Replicator::new( replicator_ledger_path, replicator_node, &leader_info, &replicator_keypair, None, ) .unwrap(); info!("started replicator.."); // Create a client which downloads from the replicator and see that it // can respond with blobs. let tn = Node::new_localhost(); let cluster_info = ClusterInfo::new(tn.info.clone()); let repair_index = replicator.entry_height(); let req = cluster_info .window_index_request_bytes(DEFAULT_SLOT_HEIGHT, repair_index) .unwrap(); let exit = Arc::new(AtomicBool::new(false)); let (s_reader, r_reader) = channel(); let repair_socket = Arc::new(tn.sockets.repair); let t_receiver = blob_receiver(repair_socket.clone(), exit.clone(), s_reader); info!( "Sending repair requests from: {} to: {}", tn.info.id, replicator_info.gossip ); let mut received_blob = false; for _ in 0..5 { repair_socket.send_to(&req, replicator_info.gossip).unwrap(); let x = r_reader.recv_timeout(Duration::new(1, 0)); if let Ok(blobs) = x { for b in blobs { let br = b.read().unwrap(); assert!(br.index() == repair_index); let entry: Entry = deserialize(&br.data()[..br.meta.size]).unwrap(); info!("entry: {:?}", entry); assert_ne!(entry.id, Hash::default()); received_blob = true; } break; } } exit.store(true, Ordering::Relaxed); t_receiver.join().unwrap(); assert!(received_blob); // The replicator will not submit storage proofs if // chacha is not enabled #[cfg(feature = "chacha")] { use solana::rpc_request::{RpcClient, RpcRequest, RpcRequestHandler}; use std::thread::sleep; info!( "looking for pubkeys for entry: {}", replicator.entry_height() ); let rpc_client = RpcClient::new_from_socket(validator_node_info.rpc); let mut non_zero_pubkeys = false; for _ in 0..60 { let params = json!([replicator.entry_height()]); let pubkeys = rpc_client .make_rpc_request(1, RpcRequest::GetStoragePubkeysForEntryHeight, Some(params)) .unwrap(); info!("pubkeys: {:?}", pubkeys); if pubkeys.as_array().unwrap().len() != 0 { non_zero_pubkeys = true; break; } sleep(Duration::from_secs(1)); } assert!(non_zero_pubkeys); } replicator.close(); validator_exit(); leader_exit(); } info!("cleanup"); Blocktree::destroy(&leader_ledger_path).expect("Expected successful database destruction"); Blocktree::destroy(&replicator_ledger_path).expect("Expected successful database destruction"); let _ignored = remove_dir_all(&leader_ledger_path); let _ignored = remove_dir_all(&replicator_ledger_path); } #[test] fn test_replicator_startup_leader_hang() { use std::net::{IpAddr, Ipv4Addr, SocketAddr}; use std::time::Duration; solana_logger::setup(); info!("starting replicator test"); let replicator_ledger_path = &get_tmp_ledger_path("replicator_test_replicator_ledger"); let leader_ledger_path = "replicator_test_leader_ledger"; { let replicator_keypair = Keypair::new(); info!("starting replicator node"); let replicator_node = Node::new_localhost_with_pubkey(replicator_keypair.pubkey()); let fake_gossip = SocketAddr::new(IpAddr::V4(Ipv4Addr::new(0, 0, 0, 0)), 0); let leader_info = NodeInfo::new_entry_point(&fake_gossip); let replicator_res = Replicator::new( replicator_ledger_path, replicator_node, &leader_info, &replicator_keypair, Some(Duration::from_secs(3)), ); assert!(replicator_res.is_err()); } let _ignored = Blocktree::destroy(&leader_ledger_path); let _ignored = Blocktree::destroy(&replicator_ledger_path); let _ignored = remove_dir_all(&leader_ledger_path); let _ignored = remove_dir_all(&replicator_ledger_path); } #[test] fn test_replicator_startup_ledger_hang() { use std::net::UdpSocket; solana_logger::setup(); info!("starting replicator test"); let replicator_ledger_path = &get_tmp_ledger_path("replicator_test_replicator_ledger"); info!("starting leader node"); let leader_keypair = Arc::new(Keypair::new()); let leader_node = Node::new_localhost_with_pubkey(leader_keypair.pubkey()); let leader_info = leader_node.info.clone(); let leader_ledger_path = "replicator_test_leader_ledger"; let fullnode_config = FullnodeConfig::default(); let blocktree_config = fullnode_config.ledger_config(); let ( _mint_keypair, leader_ledger_path, _tick_height, _last_entry_height, _last_id, _last_entry_id, ) = create_tmp_sample_ledger( leader_ledger_path, 100, 0, leader_info.id, 42, &blocktree_config, ); let validator_ledger_path = tmp_copy_ledger( &leader_ledger_path, "replicator_test_validator_ledger", &blocktree_config, ); { let voting_keypair = VotingKeypair::new_local(&leader_keypair); let _ = Fullnode::new( leader_node, &leader_keypair, &leader_ledger_path, voting_keypair, None, &fullnode_config, ); let validator_keypair = Arc::new(Keypair::new()); let voting_keypair = VotingKeypair::new_local(&validator_keypair); let validator_node = Node::new_localhost_with_pubkey(validator_keypair.pubkey()); let _ = Fullnode::new( validator_node, &validator_keypair, &validator_ledger_path, voting_keypair, Some(&leader_info), &FullnodeConfig::default(), ); info!("starting replicator node"); let bad_keys = Keypair::new(); let mut replicator_node = Node::new_localhost_with_pubkey(bad_keys.pubkey()); // Pass bad TVU sockets to prevent successful ledger download replicator_node.sockets.tvu = vec![UdpSocket::bind("0.0.0.0:0").unwrap()]; let leader_info = NodeInfo::new_entry_point(&leader_info.gossip); let replicator_res = Replicator::new( replicator_ledger_path, replicator_node, &leader_info, &bad_keys, Some(Duration::from_secs(3)), ); assert!(replicator_res.is_err()); } let _ignored = Blocktree::destroy(&leader_ledger_path); let _ignored = Blocktree::destroy(&replicator_ledger_path); let _ignored = remove_dir_all(&leader_ledger_path); let _ignored = remove_dir_all(&replicator_ledger_path); }