| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  | // Long-running ledger_cleanup tests
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[cfg(test)]
 | 
					
						
							|  |  |  | mod tests {
 | 
					
						
							|  |  |  |     use solana_core::ledger_cleanup_service::LedgerCleanupService;
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |     use solana_ledger::blockstore::{make_many_slot_entries, Blockstore};
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |     use solana_ledger::get_tmp_ledger_path;
 | 
					
						
							|  |  |  |     use solana_ledger::shred::Shred;
 | 
					
						
							|  |  |  |     use std::collections::VecDeque;
 | 
					
						
							|  |  |  |     use std::str::FromStr;
 | 
					
						
							|  |  |  |     use std::sync::atomic::{AtomicBool, Ordering};
 | 
					
						
							|  |  |  |     use std::sync::mpsc::channel;
 | 
					
						
							|  |  |  |     use std::sync::{Arc, RwLock};
 | 
					
						
							| 
									
										
										
										
											2020-04-15 11:54:03 -07:00
										 |  |  |     use std::thread::{self, Builder, JoinHandle};
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |     use std::time::{Duration, Instant};
 | 
					
						
							|  |  |  |     use systemstat::{CPULoad, Platform, System};
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-03 12:05:14 -08:00
										 |  |  |     const DEFAULT_BENCHMARK_SLOTS: u64 = 50;
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |     const DEFAULT_BATCH_SIZE: u64 = 1;
 | 
					
						
							| 
									
										
										
										
											2020-03-31 17:21:19 -07:00
										 |  |  |     const DEFAULT_MAX_LEDGER_SHREDS: u64 = 50;
 | 
					
						
							| 
									
										
										
										
											2020-01-03 12:05:14 -08:00
										 |  |  |     const DEFAULT_ENTRIES_PER_SLOT: u64 = 500;
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |     const DEFAULT_STOP_SIZE_BYTES: u64 = 0;
 | 
					
						
							|  |  |  |     const DEFAULT_STOP_SIZE_ITERATIONS: u64 = 0;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const ROCKSDB_FLUSH_GRACE_PERIOD_SECS: u64 = 20;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[derive(Debug)]
 | 
					
						
							|  |  |  |     struct BenchmarkConfig {
 | 
					
						
							|  |  |  |         pub benchmark_slots: u64,
 | 
					
						
							|  |  |  |         pub batch_size: u64,
 | 
					
						
							| 
									
										
										
										
											2020-03-31 17:21:19 -07:00
										 |  |  |         pub max_ledger_shreds: u64,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |         pub entries_per_slot: u64,
 | 
					
						
							|  |  |  |         pub stop_size_bytes: u64,
 | 
					
						
							|  |  |  |         pub stop_size_iterations: u64,
 | 
					
						
							|  |  |  |         pub pre_generate_data: bool,
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |         pub cleanup_blockstore: bool,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |         pub emit_cpu_info: bool,
 | 
					
						
							|  |  |  |         pub assert_compaction: bool,
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[derive(Clone, Copy, Debug)]
 | 
					
						
							|  |  |  |     struct CpuStatsInner {
 | 
					
						
							|  |  |  |         pub cpu_user: f32,
 | 
					
						
							|  |  |  |         pub cpu_system: f32,
 | 
					
						
							|  |  |  |         pub cpu_idle: f32,
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     impl From<CPULoad> for CpuStatsInner {
 | 
					
						
							|  |  |  |         fn from(cpu: CPULoad) -> Self {
 | 
					
						
							|  |  |  |             Self {
 | 
					
						
							|  |  |  |                 cpu_user: cpu.user * 100.0,
 | 
					
						
							|  |  |  |                 cpu_system: cpu.system * 100.0,
 | 
					
						
							|  |  |  |                 cpu_idle: cpu.idle * 100.0,
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     impl Default for CpuStatsInner {
 | 
					
						
							|  |  |  |         fn default() -> Self {
 | 
					
						
							|  |  |  |             Self {
 | 
					
						
							|  |  |  |                 cpu_user: 0.0,
 | 
					
						
							|  |  |  |                 cpu_system: 0.0,
 | 
					
						
							|  |  |  |                 cpu_idle: 0.0,
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     struct CpuStats {
 | 
					
						
							|  |  |  |         stats: RwLock<CpuStatsInner>,
 | 
					
						
							|  |  |  |         sys: System,
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     impl Default for CpuStats {
 | 
					
						
							|  |  |  |         fn default() -> Self {
 | 
					
						
							|  |  |  |             Self {
 | 
					
						
							|  |  |  |                 stats: RwLock::new(CpuStatsInner::default()),
 | 
					
						
							|  |  |  |                 sys: System::new(),
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     impl CpuStats {
 | 
					
						
							|  |  |  |         fn update(&self) {
 | 
					
						
							| 
									
										
										
										
											2020-05-15 17:35:43 +01:00
										 |  |  |             if let Ok(cpu) = self.sys.cpu_load_aggregate() {
 | 
					
						
							|  |  |  |                 std::thread::sleep(Duration::from_millis(400));
 | 
					
						
							|  |  |  |                 let cpu_new = CpuStatsInner::from(cpu.done().unwrap());
 | 
					
						
							|  |  |  |                 *self.stats.write().unwrap() = cpu_new;
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fn get_stats(&self) -> CpuStatsInner {
 | 
					
						
							| 
									
										
										
										
											2020-05-15 17:35:43 +01:00
										 |  |  |             *self.stats.read().unwrap()
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     struct CpuStatsUpdater {
 | 
					
						
							|  |  |  |         cpu_stats: Arc<CpuStats>,
 | 
					
						
							|  |  |  |         t_cleanup: JoinHandle<()>,
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     impl CpuStatsUpdater {
 | 
					
						
							|  |  |  |         pub fn new(exit: &Arc<AtomicBool>) -> Self {
 | 
					
						
							|  |  |  |             let exit = exit.clone();
 | 
					
						
							|  |  |  |             let cpu_stats = Arc::new(CpuStats::default());
 | 
					
						
							|  |  |  |             let cpu_stats_clone = cpu_stats.clone();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             let t_cleanup = Builder::new()
 | 
					
						
							|  |  |  |                 .name("cpu_info".to_string())
 | 
					
						
							|  |  |  |                 .spawn(move || loop {
 | 
					
						
							|  |  |  |                     if exit.load(Ordering::Relaxed) {
 | 
					
						
							|  |  |  |                         break;
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                     cpu_stats_clone.update();
 | 
					
						
							|  |  |  |                 })
 | 
					
						
							|  |  |  |                 .unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             Self {
 | 
					
						
							| 
									
										
										
										
											2020-05-15 17:35:43 +01:00
										 |  |  |                 cpu_stats,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |                 t_cleanup,
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         pub fn get_stats(&self) -> CpuStatsInner {
 | 
					
						
							|  |  |  |             self.cpu_stats.get_stats()
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         pub fn join(self) -> std::thread::Result<()> {
 | 
					
						
							|  |  |  |             self.t_cleanup.join()
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn read_env<T>(key: &str, default: T) -> T
 | 
					
						
							|  |  |  |     where
 | 
					
						
							|  |  |  |         T: FromStr,
 | 
					
						
							|  |  |  |     {
 | 
					
						
							|  |  |  |         match std::env::var(key) {
 | 
					
						
							|  |  |  |             Ok(val) => val.parse().unwrap_or(default),
 | 
					
						
							|  |  |  |             Err(_e) => default,
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn get_benchmark_config() -> BenchmarkConfig {
 | 
					
						
							|  |  |  |         let benchmark_slots = read_env("BENCHMARK_SLOTS", DEFAULT_BENCHMARK_SLOTS);
 | 
					
						
							|  |  |  |         let batch_size = read_env("BATCH_SIZE", DEFAULT_BATCH_SIZE);
 | 
					
						
							| 
									
										
										
										
											2020-03-31 17:21:19 -07:00
										 |  |  |         let max_ledger_shreds = read_env("MAX_LEDGER_SHREDS", DEFAULT_MAX_LEDGER_SHREDS);
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |         let entries_per_slot = read_env("ENTRIES_PER_SLOT", DEFAULT_ENTRIES_PER_SLOT);
 | 
					
						
							|  |  |  |         let stop_size_bytes = read_env("STOP_SIZE_BYTES", DEFAULT_STOP_SIZE_BYTES);
 | 
					
						
							|  |  |  |         let stop_size_iterations = read_env("STOP_SIZE_ITERATIONS", DEFAULT_STOP_SIZE_ITERATIONS);
 | 
					
						
							|  |  |  |         let pre_generate_data = read_env("PRE_GENERATE_DATA", false);
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |         let cleanup_blockstore = read_env("CLEANUP_BLOCKSTORE", true);
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |         let emit_cpu_info = read_env("EMIT_CPU_INFO", true);
 | 
					
						
							|  |  |  |         // set default to `true` once compaction is merged
 | 
					
						
							|  |  |  |         let assert_compaction = read_env("ASSERT_COMPACTION", false);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         BenchmarkConfig {
 | 
					
						
							|  |  |  |             benchmark_slots,
 | 
					
						
							|  |  |  |             batch_size,
 | 
					
						
							| 
									
										
										
										
											2020-03-31 17:21:19 -07:00
										 |  |  |             max_ledger_shreds,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |             entries_per_slot,
 | 
					
						
							|  |  |  |             stop_size_bytes,
 | 
					
						
							|  |  |  |             stop_size_iterations,
 | 
					
						
							|  |  |  |             pre_generate_data,
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |             cleanup_blockstore,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |             emit_cpu_info,
 | 
					
						
							|  |  |  |             assert_compaction,
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn emit_header() {
 | 
					
						
							|  |  |  |         println!("TIME_MS,DELTA_MS,START_SLOT,BATCH_SIZE,ENTRIES,MAX,SIZE,DELTA_SIZE,CPU_USER,CPU_SYSTEM,CPU_IDLE");
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn emit_stats(
 | 
					
						
							| 
									
										
										
										
											2020-07-28 02:33:27 -07:00
										 |  |  |         time_initial: Instant,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |         time_previous: &mut Instant,
 | 
					
						
							|  |  |  |         storage_previous: &mut u64,
 | 
					
						
							|  |  |  |         start_slot: u64,
 | 
					
						
							|  |  |  |         batch_size: u64,
 | 
					
						
							|  |  |  |         entries: u64,
 | 
					
						
							|  |  |  |         max_slots: i64,
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |         blockstore: &Blockstore,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |         cpu: &CpuStatsInner,
 | 
					
						
							|  |  |  |     ) {
 | 
					
						
							|  |  |  |         let time_now = Instant::now();
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |         let storage_now = blockstore.storage_size().unwrap_or(0);
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |         let (cpu_user, cpu_system, cpu_idle) = (cpu.cpu_user, cpu.cpu_system, cpu.cpu_idle);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         println!(
 | 
					
						
							|  |  |  |             "{},{},{},{},{},{},{},{},{},{},{}",
 | 
					
						
							| 
									
										
										
										
											2020-07-28 02:33:27 -07:00
										 |  |  |             time_now.duration_since(time_initial).as_millis(),
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |             time_now.duration_since(*time_previous).as_millis(),
 | 
					
						
							|  |  |  |             start_slot,
 | 
					
						
							|  |  |  |             batch_size,
 | 
					
						
							|  |  |  |             entries,
 | 
					
						
							|  |  |  |             max_slots,
 | 
					
						
							|  |  |  |             storage_now,
 | 
					
						
							|  |  |  |             storage_now as i64 - *storage_previous as i64,
 | 
					
						
							|  |  |  |             cpu_user,
 | 
					
						
							|  |  |  |             cpu_system,
 | 
					
						
							|  |  |  |             cpu_idle,
 | 
					
						
							|  |  |  |         );
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         *time_previous = time_now;
 | 
					
						
							|  |  |  |         *storage_previous = storage_now;
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[test]
 | 
					
						
							|  |  |  |     fn test_ledger_cleanup_compaction() {
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |         let blockstore_path = get_tmp_ledger_path!();
 | 
					
						
							|  |  |  |         let blockstore = Arc::new(Blockstore::open(&blockstore_path).unwrap());
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |         let config = get_benchmark_config();
 | 
					
						
							|  |  |  |         eprintln!("BENCHMARK CONFIG: {:?}", config);
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |         eprintln!("LEDGER_PATH: {:?}", &blockstore_path);
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         let benchmark_slots = config.benchmark_slots;
 | 
					
						
							|  |  |  |         let batch_size = config.batch_size;
 | 
					
						
							| 
									
										
										
										
											2020-03-31 17:21:19 -07:00
										 |  |  |         let max_ledger_shreds = config.max_ledger_shreds;
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |         let entries_per_slot = config.entries_per_slot;
 | 
					
						
							|  |  |  |         let stop_size_bytes = config.stop_size_bytes;
 | 
					
						
							|  |  |  |         let stop_size_iterations = config.stop_size_iterations;
 | 
					
						
							|  |  |  |         let pre_generate_data = config.pre_generate_data;
 | 
					
						
							|  |  |  |         let batches = benchmark_slots / batch_size;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let (sender, receiver) = channel();
 | 
					
						
							|  |  |  |         let exit = Arc::new(AtomicBool::new(false));
 | 
					
						
							|  |  |  |         let cleaner =
 | 
					
						
							| 
									
										
										
										
											2020-03-31 17:21:19 -07:00
										 |  |  |             LedgerCleanupService::new(receiver, blockstore.clone(), max_ledger_shreds, &exit);
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         let exit_cpu = Arc::new(AtomicBool::new(false));
 | 
					
						
							|  |  |  |         let sys = CpuStatsUpdater::new(&exit_cpu);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let mut generated_batches = VecDeque::<Vec<Shred>>::new();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if pre_generate_data {
 | 
					
						
							|  |  |  |             let t0 = Instant::now();
 | 
					
						
							|  |  |  |             eprintln!("PRE_GENERATE_DATA: (this may take a while)");
 | 
					
						
							|  |  |  |             for i in 0..batches {
 | 
					
						
							|  |  |  |                 let x = i * batch_size;
 | 
					
						
							|  |  |  |                 let (shreds, _) = make_many_slot_entries(x, batch_size, entries_per_slot);
 | 
					
						
							|  |  |  |                 generated_batches.push_back(shreds);
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             eprintln!("PRE_GENERATE_DATA: took {} ms", t0.elapsed().as_millis());
 | 
					
						
							|  |  |  |         };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let time_initial = Instant::now();
 | 
					
						
							|  |  |  |         let mut time_previous = time_initial;
 | 
					
						
							|  |  |  |         let mut storage_previous = 0;
 | 
					
						
							|  |  |  |         let mut stop_size_bytes_exceeded_iterations = 0;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         emit_header();
 | 
					
						
							|  |  |  |         emit_stats(
 | 
					
						
							| 
									
										
										
										
											2020-07-28 02:33:27 -07:00
										 |  |  |             time_initial,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |             &mut time_previous,
 | 
					
						
							|  |  |  |             &mut storage_previous,
 | 
					
						
							|  |  |  |             0,
 | 
					
						
							|  |  |  |             0,
 | 
					
						
							|  |  |  |             0,
 | 
					
						
							|  |  |  |             0,
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |             &blockstore,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |             &sys.get_stats(),
 | 
					
						
							|  |  |  |         );
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for i in 0..batches {
 | 
					
						
							|  |  |  |             let x = i * batch_size;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             let shreds = if pre_generate_data {
 | 
					
						
							|  |  |  |                 generated_batches.pop_front().unwrap()
 | 
					
						
							|  |  |  |             } else {
 | 
					
						
							|  |  |  |                 make_many_slot_entries(x, batch_size, entries_per_slot).0
 | 
					
						
							|  |  |  |             };
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |             blockstore.insert_shreds(shreds, None, false).unwrap();
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |             sender.send(x).unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             emit_stats(
 | 
					
						
							| 
									
										
										
										
											2020-07-28 02:33:27 -07:00
										 |  |  |                 time_initial,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |                 &mut time_previous,
 | 
					
						
							|  |  |  |                 &mut storage_previous,
 | 
					
						
							|  |  |  |                 x,
 | 
					
						
							|  |  |  |                 batch_size,
 | 
					
						
							|  |  |  |                 batch_size,
 | 
					
						
							| 
									
										
										
										
											2020-03-31 17:21:19 -07:00
										 |  |  |                 max_ledger_shreds as i64,
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |                 &blockstore,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |                 &sys.get_stats(),
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if stop_size_bytes > 0 {
 | 
					
						
							|  |  |  |                 if storage_previous >= stop_size_bytes {
 | 
					
						
							|  |  |  |                     stop_size_bytes_exceeded_iterations += 1;
 | 
					
						
							|  |  |  |                 } else {
 | 
					
						
							|  |  |  |                     stop_size_bytes_exceeded_iterations = 0;
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if stop_size_bytes_exceeded_iterations > stop_size_iterations {
 | 
					
						
							|  |  |  |                     break;
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let u1 = storage_previous;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // send final `ledger_cleanup` notification (since iterations above are zero-based)
 | 
					
						
							|  |  |  |         sender.send(benchmark_slots).unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         emit_stats(
 | 
					
						
							| 
									
										
										
										
											2020-07-28 02:33:27 -07:00
										 |  |  |             time_initial,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |             &mut time_previous,
 | 
					
						
							|  |  |  |             &mut storage_previous,
 | 
					
						
							|  |  |  |             benchmark_slots,
 | 
					
						
							|  |  |  |             0,
 | 
					
						
							|  |  |  |             0,
 | 
					
						
							| 
									
										
										
										
											2020-03-31 17:21:19 -07:00
										 |  |  |             max_ledger_shreds as i64,
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |             &blockstore,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |             &sys.get_stats(),
 | 
					
						
							|  |  |  |         );
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-03 12:05:14 -08:00
										 |  |  |         // Poll on some compaction happening
 | 
					
						
							|  |  |  |         let start_poll = Instant::now();
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |         while blockstore.storage_size().unwrap_or(0) >= u1 {
 | 
					
						
							| 
									
										
										
										
											2020-01-03 12:05:14 -08:00
										 |  |  |             if start_poll.elapsed().as_secs() > ROCKSDB_FLUSH_GRACE_PERIOD_SECS {
 | 
					
						
							|  |  |  |                 break;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             std::thread::sleep(Duration::from_millis(200));
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         emit_stats(
 | 
					
						
							| 
									
										
										
										
											2020-07-28 02:33:27 -07:00
										 |  |  |             time_initial,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |             &mut time_previous,
 | 
					
						
							|  |  |  |             &mut storage_previous,
 | 
					
						
							|  |  |  |             benchmark_slots,
 | 
					
						
							|  |  |  |             0,
 | 
					
						
							|  |  |  |             0,
 | 
					
						
							| 
									
										
										
										
											2020-03-31 17:21:19 -07:00
										 |  |  |             max_ledger_shreds as i64,
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |             &blockstore,
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |             &sys.get_stats(),
 | 
					
						
							|  |  |  |         );
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let u2 = storage_previous;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         exit.store(true, Ordering::SeqCst);
 | 
					
						
							|  |  |  |         cleaner.join().unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         exit_cpu.store(true, Ordering::SeqCst);
 | 
					
						
							|  |  |  |         sys.join().unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if config.assert_compaction {
 | 
					
						
							|  |  |  |             assert!(u2 < u1, "expected compaction! pre={},post={}", u1, u2);
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-13 14:13:52 -07:00
										 |  |  |         if config.cleanup_blockstore {
 | 
					
						
							|  |  |  |             drop(blockstore);
 | 
					
						
							|  |  |  |             Blockstore::destroy(&blockstore_path)
 | 
					
						
							|  |  |  |                 .expect("Expected successful database destruction");
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2020-04-15 11:54:03 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     #[test]
 | 
					
						
							|  |  |  |     fn test_compaction() {
 | 
					
						
							|  |  |  |         let blockstore_path = get_tmp_ledger_path!();
 | 
					
						
							|  |  |  |         let blockstore = Arc::new(Blockstore::open(&blockstore_path).unwrap());
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let n = 10_000;
 | 
					
						
							|  |  |  |         let batch_size = 100;
 | 
					
						
							|  |  |  |         let batches = n / batch_size;
 | 
					
						
							|  |  |  |         let max_ledger_shreds = 100;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for i in 0..batches {
 | 
					
						
							|  |  |  |             let (shreds, _) = make_many_slot_entries(i * batch_size, batch_size, 1);
 | 
					
						
							|  |  |  |             blockstore.insert_shreds(shreds, None, false).unwrap();
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let u1 = blockstore.storage_size().unwrap() as f64;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // send signal to cleanup slots
 | 
					
						
							|  |  |  |         let (sender, receiver) = channel();
 | 
					
						
							|  |  |  |         sender.send(n).unwrap();
 | 
					
						
							| 
									
										
										
										
											2020-06-04 21:06:06 -07:00
										 |  |  |         let mut last_purge_slot = 0;
 | 
					
						
							|  |  |  |         let mut last_compaction_slot = 0;
 | 
					
						
							| 
									
										
										
										
											2020-04-15 11:54:03 -07:00
										 |  |  |         LedgerCleanupService::cleanup_ledger(
 | 
					
						
							|  |  |  |             &receiver,
 | 
					
						
							|  |  |  |             &blockstore,
 | 
					
						
							|  |  |  |             max_ledger_shreds,
 | 
					
						
							| 
									
										
										
										
											2020-06-04 21:06:06 -07:00
										 |  |  |             &mut last_purge_slot,
 | 
					
						
							| 
									
										
										
										
											2020-04-15 11:54:03 -07:00
										 |  |  |             10,
 | 
					
						
							| 
									
										
										
										
											2020-06-04 21:06:06 -07:00
										 |  |  |             &mut last_compaction_slot,
 | 
					
						
							|  |  |  |             10,
 | 
					
						
							| 
									
										
										
										
											2020-04-15 11:54:03 -07:00
										 |  |  |         )
 | 
					
						
							|  |  |  |         .unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         thread::sleep(Duration::from_secs(2));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let u2 = blockstore.storage_size().unwrap() as f64;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         assert!(u2 < u1, "insufficient compaction! pre={},post={}", u1, u2,);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // check that early slots don't exist
 | 
					
						
							|  |  |  |         let max_slot = n - max_ledger_shreds - 1;
 | 
					
						
							|  |  |  |         blockstore
 | 
					
						
							|  |  |  |             .slot_meta_iterator(0)
 | 
					
						
							|  |  |  |             .unwrap()
 | 
					
						
							|  |  |  |             .for_each(|(slot, _)| assert!(slot > max_slot));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         drop(blockstore);
 | 
					
						
							|  |  |  |         Blockstore::destroy(&blockstore_path).expect("Expected successful database destruction");
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2019-12-18 18:31:04 -05:00
										 |  |  | }
 |