| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  | #[macro_use]
 | 
					
						
							|  |  |  | extern crate log;
 | 
					
						
							|  |  |  | extern crate rayon;
 | 
					
						
							|  |  |  | extern crate solana;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use rayon::iter::*;
 | 
					
						
							| 
									
										
										
										
											2018-10-08 20:55:54 -06:00
										 |  |  | use solana::cluster_info::{ClusterInfo, Node};
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  | use solana::logger;
 | 
					
						
							| 
									
										
										
										
											2018-06-07 16:06:32 -06:00
										 |  |  | use solana::ncp::Ncp;
 | 
					
						
							| 
									
										
										
										
											2018-09-26 16:50:12 +00:00
										 |  |  | use solana::packet::{Blob, SharedBlob};
 | 
					
						
							| 
									
										
										
										
											2018-08-05 21:40:23 -07:00
										 |  |  | use solana::result;
 | 
					
						
							| 
									
										
										
										
											2018-07-03 23:21:53 -06:00
										 |  |  | use solana::service::Service;
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  | use std::net::UdpSocket;
 | 
					
						
							|  |  |  | use std::sync::atomic::{AtomicBool, Ordering};
 | 
					
						
							|  |  |  | use std::sync::{Arc, RwLock};
 | 
					
						
							|  |  |  | use std::thread::sleep;
 | 
					
						
							|  |  |  | use std::time::Duration;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-08 20:55:54 -06:00
										 |  |  | fn test_node(exit: Arc<AtomicBool>) -> (Arc<RwLock<ClusterInfo>>, Ncp, UdpSocket) {
 | 
					
						
							| 
									
										
										
										
											2018-09-14 16:56:06 -07:00
										 |  |  |     let mut tn = Node::new_localhost();
 | 
					
						
							| 
									
										
										
										
											2018-10-08 20:55:54 -06:00
										 |  |  |     let cluster_info = ClusterInfo::new(tn.info.clone()).expect("ClusterInfo::new");
 | 
					
						
							|  |  |  |     let c = Arc::new(RwLock::new(cluster_info));
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |     let w = Arc::new(RwLock::new(vec![]));
 | 
					
						
							| 
									
										
										
										
											2018-09-18 08:02:57 -07:00
										 |  |  |     let d = Ncp::new(&c.clone(), w, None, tn.sockets.gossip, exit);
 | 
					
						
							| 
									
										
										
										
											2018-09-14 16:56:06 -07:00
										 |  |  |     (c, d, tn.sockets.replicate.pop().unwrap())
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Test that the network converges.
 | 
					
						
							| 
									
										
										
										
											2018-07-11 00:18:48 -07:00
										 |  |  | /// Run until every node in the network has a full NodeInfo set.
 | 
					
						
							|  |  |  | /// Check that nodes stop sending updates after all the NodeInfo has been shared.
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  | /// tests that actually use this function are below
 | 
					
						
							|  |  |  | fn run_gossip_topo<F>(topo: F)
 | 
					
						
							|  |  |  | where
 | 
					
						
							| 
									
										
										
										
											2018-10-08 20:55:54 -06:00
										 |  |  |     F: Fn(&Vec<(Arc<RwLock<ClusterInfo>>, Ncp, UdpSocket)>) -> (),
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  | {
 | 
					
						
							|  |  |  |     let num: usize = 5;
 | 
					
						
							|  |  |  |     let exit = Arc::new(AtomicBool::new(false));
 | 
					
						
							|  |  |  |     let listen: Vec<_> = (0..num).map(|_| test_node(exit.clone())).collect();
 | 
					
						
							|  |  |  |     topo(&listen);
 | 
					
						
							|  |  |  |     let mut done = true;
 | 
					
						
							|  |  |  |     for i in 0..(num * 32) {
 | 
					
						
							|  |  |  |         done = false;
 | 
					
						
							|  |  |  |         trace!("round {}", i);
 | 
					
						
							| 
									
										
										
										
											2018-06-04 17:17:23 -06:00
										 |  |  |         for (c, _, _) in &listen {
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |             if num == c.read().unwrap().convergence() as usize {
 | 
					
						
							|  |  |  |                 done = true;
 | 
					
						
							|  |  |  |                 break;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         //at least 1 node converged
 | 
					
						
							|  |  |  |         if done == true {
 | 
					
						
							|  |  |  |             break;
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         sleep(Duration::new(1, 0));
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     exit.store(true, Ordering::Relaxed);
 | 
					
						
							| 
									
										
										
										
											2018-07-03 23:21:53 -06:00
										 |  |  |     for (c, dr, _) in listen {
 | 
					
						
							|  |  |  |         dr.join().unwrap();
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |         // make it clear what failed
 | 
					
						
							|  |  |  |         // protocol is to chatty, updates should stop after everyone receives `num`
 | 
					
						
							|  |  |  |         assert!(c.read().unwrap().update_index <= num as u64);
 | 
					
						
							|  |  |  |         // protocol is not chatty enough, everyone should get `num` entries
 | 
					
						
							|  |  |  |         assert_eq!(c.read().unwrap().table.len(), num);
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     assert!(done);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | /// ring a -> b -> c -> d -> e -> a
 | 
					
						
							|  |  |  | #[test]
 | 
					
						
							| 
									
										
										
										
											2018-08-05 21:40:23 -07:00
										 |  |  | fn gossip_ring() -> result::Result<()> {
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |     logger::setup();
 | 
					
						
							|  |  |  |     run_gossip_topo(|listen| {
 | 
					
						
							|  |  |  |         let num = listen.len();
 | 
					
						
							|  |  |  |         for n in 0..num {
 | 
					
						
							|  |  |  |             let y = n % listen.len();
 | 
					
						
							|  |  |  |             let x = (n + 1) % listen.len();
 | 
					
						
							|  |  |  |             let mut xv = listen[x].0.write().unwrap();
 | 
					
						
							|  |  |  |             let yv = listen[y].0.read().unwrap();
 | 
					
						
							| 
									
										
										
										
											2018-08-30 12:07:54 -07:00
										 |  |  |             let mut d = yv.table[&yv.id].clone();
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |             d.version = 0;
 | 
					
						
							|  |  |  |             xv.insert(&d);
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     });
 | 
					
						
							| 
									
										
										
										
											2018-08-05 21:40:23 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Ok(())
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// star a -> (b,c,d,e)
 | 
					
						
							|  |  |  | #[test]
 | 
					
						
							|  |  |  | fn gossip_star() {
 | 
					
						
							|  |  |  |     logger::setup();
 | 
					
						
							|  |  |  |     run_gossip_topo(|listen| {
 | 
					
						
							|  |  |  |         let num = listen.len();
 | 
					
						
							|  |  |  |         for n in 0..(num - 1) {
 | 
					
						
							|  |  |  |             let x = 0;
 | 
					
						
							|  |  |  |             let y = (n + 1) % listen.len();
 | 
					
						
							|  |  |  |             let mut xv = listen[x].0.write().unwrap();
 | 
					
						
							|  |  |  |             let yv = listen[y].0.read().unwrap();
 | 
					
						
							| 
									
										
										
										
											2018-08-30 12:07:54 -07:00
										 |  |  |             let mut yd = yv.table[&yv.id].clone();
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |             yd.version = 0;
 | 
					
						
							|  |  |  |             xv.insert(&yd);
 | 
					
						
							| 
									
										
										
										
											2018-08-30 12:07:54 -07:00
										 |  |  |             trace!("star leader {:?}", &xv.id.as_ref()[..4]);
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |         }
 | 
					
						
							|  |  |  |     });
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// rstar a <- (b,c,d,e)
 | 
					
						
							|  |  |  | #[test]
 | 
					
						
							|  |  |  | fn gossip_rstar() {
 | 
					
						
							|  |  |  |     logger::setup();
 | 
					
						
							|  |  |  |     run_gossip_topo(|listen| {
 | 
					
						
							|  |  |  |         let num = listen.len();
 | 
					
						
							|  |  |  |         let xd = {
 | 
					
						
							|  |  |  |             let xv = listen[0].0.read().unwrap();
 | 
					
						
							| 
									
										
										
										
											2018-08-30 12:07:54 -07:00
										 |  |  |             xv.table[&xv.id].clone()
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |         };
 | 
					
						
							| 
									
										
										
										
											2018-07-31 15:50:09 -06:00
										 |  |  |         trace!("rstar leader {:?}", &xd.id.as_ref()[..4]);
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |         for n in 0..(num - 1) {
 | 
					
						
							|  |  |  |             let y = (n + 1) % listen.len();
 | 
					
						
							|  |  |  |             let mut yv = listen[y].0.write().unwrap();
 | 
					
						
							|  |  |  |             yv.insert(&xd);
 | 
					
						
							| 
									
										
										
										
											2018-07-31 15:50:09 -06:00
										 |  |  |             trace!(
 | 
					
						
							|  |  |  |                 "rstar insert {:?} into {:?}",
 | 
					
						
							|  |  |  |                 &xd.id.as_ref()[..4],
 | 
					
						
							| 
									
										
										
										
											2018-08-30 12:07:54 -07:00
										 |  |  |                 &yv.id.as_ref()[..4]
 | 
					
						
							| 
									
										
										
										
											2018-07-31 15:50:09 -06:00
										 |  |  |             );
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |         }
 | 
					
						
							|  |  |  |     });
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[test]
 | 
					
						
							| 
									
										
										
										
											2018-10-08 20:55:54 -06:00
										 |  |  | pub fn cluster_info_retransmit() -> result::Result<()> {
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |     logger::setup();
 | 
					
						
							|  |  |  |     let exit = Arc::new(AtomicBool::new(false));
 | 
					
						
							|  |  |  |     trace!("c1:");
 | 
					
						
							|  |  |  |     let (c1, dr1, tn1) = test_node(exit.clone());
 | 
					
						
							|  |  |  |     trace!("c2:");
 | 
					
						
							|  |  |  |     let (c2, dr2, tn2) = test_node(exit.clone());
 | 
					
						
							|  |  |  |     trace!("c3:");
 | 
					
						
							|  |  |  |     let (c3, dr3, tn3) = test_node(exit.clone());
 | 
					
						
							|  |  |  |     let c1_data = c1.read().unwrap().my_data().clone();
 | 
					
						
							|  |  |  |     c1.write().unwrap().set_leader(c1_data.id);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     c2.write().unwrap().insert(&c1_data);
 | 
					
						
							|  |  |  |     c3.write().unwrap().insert(&c1_data);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     c2.write().unwrap().set_leader(c1_data.id);
 | 
					
						
							|  |  |  |     c3.write().unwrap().set_leader(c1_data.id);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     //wait to converge
 | 
					
						
							|  |  |  |     trace!("waiting to converge:");
 | 
					
						
							|  |  |  |     let mut done = false;
 | 
					
						
							|  |  |  |     for _ in 0..30 {
 | 
					
						
							| 
									
										
										
										
											2018-06-22 22:23:52 -07:00
										 |  |  |         done = c1.read().unwrap().table.len() == 3
 | 
					
						
							|  |  |  |             && c2.read().unwrap().table.len() == 3
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |             && c3.read().unwrap().table.len() == 3;
 | 
					
						
							|  |  |  |         if done {
 | 
					
						
							|  |  |  |             break;
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         sleep(Duration::new(1, 0));
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     assert!(done);
 | 
					
						
							| 
									
										
										
										
											2018-09-26 16:50:12 +00:00
										 |  |  |     let b = SharedBlob::default();
 | 
					
						
							|  |  |  |     b.write().unwrap().meta.size = 10;
 | 
					
						
							| 
									
										
										
										
											2018-10-08 20:55:54 -06:00
										 |  |  |     ClusterInfo::retransmit(&c1, &b, &tn1)?;
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |     let res: Vec<_> = [tn1, tn2, tn3]
 | 
					
						
							|  |  |  |         .into_par_iter()
 | 
					
						
							|  |  |  |         .map(|s| {
 | 
					
						
							|  |  |  |             let mut b = Blob::default();
 | 
					
						
							|  |  |  |             s.set_read_timeout(Some(Duration::new(1, 0))).unwrap();
 | 
					
						
							|  |  |  |             let res = s.recv_from(&mut b.data);
 | 
					
						
							|  |  |  |             res.is_err() //true if failed to receive the retransmit packet
 | 
					
						
							| 
									
										
										
										
											2018-09-14 16:25:14 -07:00
										 |  |  |         }).collect();
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  |     //true if failed receive the retransmit packet, r2, and r3 should succeed
 | 
					
						
							|  |  |  |     //r1 was the sender, so it should fail to receive the packet
 | 
					
						
							|  |  |  |     assert_eq!(res, [true, false, false]);
 | 
					
						
							|  |  |  |     exit.store(true, Ordering::Relaxed);
 | 
					
						
							| 
									
										
										
										
											2018-07-03 23:21:53 -06:00
										 |  |  |     dr1.join().unwrap();
 | 
					
						
							|  |  |  |     dr2.join().unwrap();
 | 
					
						
							|  |  |  |     dr3.join().unwrap();
 | 
					
						
							| 
									
										
										
										
											2018-08-05 21:40:23 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Ok(())
 | 
					
						
							| 
									
										
										
										
											2018-05-30 09:50:28 -07:00
										 |  |  | }
 | 
					
						
							| 
									
										
										
										
											2018-06-18 23:50:41 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #[test]
 | 
					
						
							| 
									
										
										
										
											2018-07-09 17:35:23 -07:00
										 |  |  | #[ignore]
 | 
					
						
							| 
									
										
										
										
											2018-06-26 01:08:19 -07:00
										 |  |  | fn test_external_liveness_table() {
 | 
					
						
							| 
									
										
										
										
											2018-06-18 23:50:41 -07:00
										 |  |  |     logger::setup();
 | 
					
						
							| 
									
										
										
										
											2018-08-25 21:09:18 -07:00
										 |  |  |     assert!(cfg!(feature = "test"));
 | 
					
						
							| 
									
										
										
										
											2018-06-18 23:50:41 -07:00
										 |  |  |     let c1_c4_exit = Arc::new(AtomicBool::new(false));
 | 
					
						
							|  |  |  |     let c2_c3_exit = Arc::new(AtomicBool::new(false));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     trace!("c1:");
 | 
					
						
							|  |  |  |     let (c1, dr1, _) = test_node(c1_c4_exit.clone());
 | 
					
						
							|  |  |  |     trace!("c2:");
 | 
					
						
							|  |  |  |     let (c2, dr2, _) = test_node(c2_c3_exit.clone());
 | 
					
						
							|  |  |  |     trace!("c3:");
 | 
					
						
							|  |  |  |     let (c3, dr3, _) = test_node(c2_c3_exit.clone());
 | 
					
						
							|  |  |  |     trace!("c4:");
 | 
					
						
							|  |  |  |     let (c4, dr4, _) = test_node(c1_c4_exit.clone());
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let c1_data = c1.read().unwrap().my_data().clone();
 | 
					
						
							|  |  |  |     c1.write().unwrap().set_leader(c1_data.id);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-30 12:07:54 -07:00
										 |  |  |     let c2_id = c2.read().unwrap().id;
 | 
					
						
							|  |  |  |     let c3_id = c3.read().unwrap().id;
 | 
					
						
							|  |  |  |     let c4_id = c4.read().unwrap().id;
 | 
					
						
							| 
									
										
										
										
											2018-06-18 23:50:41 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Insert the remote data about c4
 | 
					
						
							|  |  |  |     let c2_index_for_c4 = 10;
 | 
					
						
							|  |  |  |     c2.write().unwrap().remote.insert(c4_id, c2_index_for_c4);
 | 
					
						
							|  |  |  |     let c3_index_for_c4 = 20;
 | 
					
						
							|  |  |  |     c3.write().unwrap().remote.insert(c4_id, c3_index_for_c4);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set up the initial network topology
 | 
					
						
							|  |  |  |     c2.write().unwrap().insert(&c1_data);
 | 
					
						
							|  |  |  |     c3.write().unwrap().insert(&c1_data);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     c2.write().unwrap().set_leader(c1_data.id);
 | 
					
						
							|  |  |  |     c3.write().unwrap().set_leader(c1_data.id);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Wait to converge
 | 
					
						
							|  |  |  |     trace!("waiting to converge:");
 | 
					
						
							|  |  |  |     let mut done = false;
 | 
					
						
							|  |  |  |     for _ in 0..30 {
 | 
					
						
							| 
									
										
										
										
											2018-06-26 01:08:19 -07:00
										 |  |  |         done = c1.read().unwrap().table.len() == 3
 | 
					
						
							|  |  |  |             && c2.read().unwrap().table.len() == 3
 | 
					
						
							| 
									
										
										
										
											2018-06-18 23:50:41 -07:00
										 |  |  |             && c3.read().unwrap().table.len() == 3;
 | 
					
						
							|  |  |  |         if done {
 | 
					
						
							|  |  |  |             break;
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         sleep(Duration::new(1, 0));
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     assert!(done);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Validate c1's external liveness table, then release lock rc1
 | 
					
						
							|  |  |  |     {
 | 
					
						
							|  |  |  |         let rc1 = c1.read().unwrap();
 | 
					
						
							| 
									
										
										
										
											2018-08-30 12:07:54 -07:00
										 |  |  |         let el = rc1.get_external_liveness_entry(&c4.read().unwrap().id);
 | 
					
						
							| 
									
										
										
										
											2018-06-18 23:50:41 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Make sure liveness table entry for c4 exists on node c1
 | 
					
						
							|  |  |  |         assert!(el.is_some());
 | 
					
						
							|  |  |  |         let liveness_map = el.unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Make sure liveness table entry contains correct result for c2
 | 
					
						
							|  |  |  |         let c2_index_result_for_c4 = liveness_map.get(&c2_id);
 | 
					
						
							|  |  |  |         assert!(c2_index_result_for_c4.is_some());
 | 
					
						
							| 
									
										
										
										
											2018-06-26 01:08:19 -07:00
										 |  |  |         assert_eq!(*(c2_index_result_for_c4.unwrap()), c2_index_for_c4);
 | 
					
						
							| 
									
										
										
										
											2018-06-18 23:50:41 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Make sure liveness table entry contains correct result for c3
 | 
					
						
							|  |  |  |         let c3_index_result_for_c4 = liveness_map.get(&c3_id);
 | 
					
						
							|  |  |  |         assert!(c3_index_result_for_c4.is_some());
 | 
					
						
							| 
									
										
										
										
											2018-06-26 01:08:19 -07:00
										 |  |  |         assert_eq!(*(c3_index_result_for_c4.unwrap()), c3_index_for_c4);
 | 
					
						
							| 
									
										
										
										
											2018-06-18 23:50:41 -07:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Shutdown validators c2 and c3
 | 
					
						
							|  |  |  |     c2_c3_exit.store(true, Ordering::Relaxed);
 | 
					
						
							| 
									
										
										
										
											2018-07-03 23:21:53 -06:00
										 |  |  |     dr2.join().unwrap();
 | 
					
						
							|  |  |  |     dr3.join().unwrap();
 | 
					
						
							| 
									
										
										
										
											2018-06-18 23:50:41 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Allow communication between c1 and c4, make sure that c1's external_liveness table
 | 
					
						
							|  |  |  |     // entry for c4 gets cleared
 | 
					
						
							|  |  |  |     c4.write().unwrap().insert(&c1_data);
 | 
					
						
							|  |  |  |     c4.write().unwrap().set_leader(c1_data.id);
 | 
					
						
							|  |  |  |     for _ in 0..30 {
 | 
					
						
							| 
									
										
										
										
											2018-08-03 11:27:44 -07:00
										 |  |  |         done = c1
 | 
					
						
							|  |  |  |             .read()
 | 
					
						
							| 
									
										
										
										
											2018-06-26 01:08:19 -07:00
										 |  |  |             .unwrap()
 | 
					
						
							|  |  |  |             .get_external_liveness_entry(&c4_id)
 | 
					
						
							|  |  |  |             .is_none();
 | 
					
						
							| 
									
										
										
										
											2018-06-18 23:50:41 -07:00
										 |  |  |         if done {
 | 
					
						
							|  |  |  |             break;
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         sleep(Duration::new(1, 0));
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     assert!(done);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Shutdown validators c1 and c4
 | 
					
						
							|  |  |  |     c1_c4_exit.store(true, Ordering::Relaxed);
 | 
					
						
							| 
									
										
										
										
											2018-07-03 23:21:53 -06:00
										 |  |  |     dr1.join().unwrap();
 | 
					
						
							|  |  |  |     dr4.join().unwrap();
 | 
					
						
							| 
									
										
										
										
											2018-06-26 01:08:19 -07:00
										 |  |  | }
 |