diff --git a/src/bin/fullnode.rs b/src/bin/fullnode.rs index 33d14ab2b64e6c..90fa50f087d705 100644 --- a/src/bin/fullnode.rs +++ b/src/bin/fullnode.rs @@ -6,17 +6,13 @@ extern crate log; extern crate serde_json; #[macro_use] extern crate solana; -extern crate rocksdb; extern crate solana_metrics; use clap::{App, Arg}; -use rocksdb::{Options, DB}; use solana::client::mk_client; use solana::cluster_info::{Node, FULLNODE_PORT_RANGE}; -use solana::db_ledger::{write_entries_to_ledger, DB_LEDGER_DIRECTORY}; use solana::fullnode::{Config, Fullnode, FullnodeReturnType}; use solana::leader_scheduler::LeaderScheduler; -use solana::ledger::read_ledger; use solana::logger; use solana::netutil::find_available_port_in_range; use solana::signature::{Keypair, KeypairUtil}; @@ -96,18 +92,6 @@ fn main() { let ledger_path = matches.value_of("ledger").unwrap(); - // Create the RocksDb ledger, eventually will simply repurpose the input - // ledger path as the RocksDb ledger path - let db_ledger_path = format!("{}/{}", ledger_path, DB_LEDGER_DIRECTORY); - // Destroy old ledger - DB::destroy(&Options::default(), &db_ledger_path) - .expect("Expected successful database destruction"); - let ledger_entries: Vec<_> = read_ledger(ledger_path, true) - .expect("opening ledger") - .map(|entry| entry.unwrap()) - .collect(); - write_entries_to_ledger(&[db_ledger_path], &ledger_entries[..]); - // socketaddr that is initial pointer into the network's gossip (ncp) let network = matches .value_of("network") diff --git a/src/bin/replicator.rs b/src/bin/replicator.rs index e8c4030a7767d1..6a8ba9450106c1 100644 --- a/src/bin/replicator.rs +++ b/src/bin/replicator.rs @@ -10,7 +10,7 @@ use clap::{App, Arg}; use solana::chacha::{chacha_cbc_encrypt_file, CHACHA_BLOCK_SIZE}; use solana::client::mk_client; use solana::cluster_info::Node; -use solana::db_ledger::{DbLedger, DB_LEDGER_DIRECTORY}; +use solana::db_ledger::DbLedger; use solana::fullnode::Config; use solana::ledger::LEDGER_DATA_FILE; use solana::logger; @@ -97,9 +97,8 @@ fn main() { // Create the RocksDb ledger, eventually will simply repurpose the input // ledger path as the RocksDb ledger path - let db_ledger_path = format!("{}/{}", ledger_path.unwrap(), DB_LEDGER_DIRECTORY); let db_ledger = Arc::new(RwLock::new( - DbLedger::open(&db_ledger_path).expect("Expected to be able to open database ledger"), + DbLedger::open(&ledger_path.unwrap()).expect("Expected to be able to open database ledger"), )); let (replicator, leader_info) = Replicator::new( db_ledger, diff --git a/src/db_ledger.rs b/src/db_ledger.rs index 244c3bd680269e..3d0c7ab4310c80 100644 --- a/src/db_ledger.rs +++ b/src/db_ledger.rs @@ -234,6 +234,8 @@ pub const ERASURE_CF: &str = "erasure"; impl DbLedger { // Opens a Ledger in directory, provides "infinite" window of blobs pub fn open(ledger_path: &str) -> Result { + let ledger_path = format!("{}/{}", ledger_path, DB_LEDGER_DIRECTORY); + // Use default database options let mut options = Options::default(); options.create_if_missing(true); @@ -262,6 +264,12 @@ impl DbLedger { }) } + pub fn destroy(ledger_path: &str) -> Result<()> { + let ledger_path = format!("{}/{}", ledger_path, DB_LEDGER_DIRECTORY); + DB::destroy(&Options::default(), &ledger_path)?; + Ok(()) + } + pub fn write_shared_blobs(&mut self, slot: u64, shared_blobs: I) -> Result<()> where I: IntoIterator, @@ -289,13 +297,14 @@ impl DbLedger { Ok(()) } - pub fn write_entries<'a, I>(&mut self, slot: u64, entries: I) -> Result<()> + pub fn write_entries(&mut self, slot: u64, entries: I) -> Result<()> where - I: IntoIterator, + I: IntoIterator, + I::Item: Borrow, { let default_addr = SocketAddr::new(IpAddr::V4(Ipv4Addr::new(0, 0, 0, 0)), 0); let shared_blobs = entries.into_iter().enumerate().map(|(idx, entry)| { - entry.to_blob( + entry.borrow().to_blob( Some(idx as u64), Some(Pubkey::default()), Some(&default_addr), @@ -439,15 +448,17 @@ impl DbLedger { } } -pub fn write_entries_to_ledger<'a, I>(ledger_paths: &[String], entries: I) +pub fn write_entries_to_ledger(ledger_paths: &[&str], entries: I) where - I: IntoIterator + Copy, + I: IntoIterator, + I::Item: Borrow, { + let mut entries = entries.into_iter(); for ledger_path in ledger_paths { let mut db_ledger = DbLedger::open(ledger_path).expect("Expected to be able to open database ledger"); db_ledger - .write_entries(DEFAULT_SLOT_HEIGHT, entries) + .write_entries(DEFAULT_SLOT_HEIGHT, entries.by_ref()) .expect("Expected successful write of genesis entries"); } } @@ -456,7 +467,6 @@ where mod tests { use super::*; use ledger::{get_tmp_ledger_path, make_tiny_test_entries, Block}; - use rocksdb::{Options, DB}; #[test] fn test_put_get_simple() { @@ -506,8 +516,7 @@ mod tests { // Destroying database without closing it first is undefined behavior drop(ledger); - DB::destroy(&Options::default(), &ledger_path) - .expect("Expected successful database destruction"); + DbLedger::destroy(&ledger_path).expect("Expected successful database destruction"); } #[test] @@ -569,8 +578,7 @@ mod tests { // Destroying database without closing it first is undefined behavior drop(ledger); - DB::destroy(&Options::default(), &ledger_path) - .expect("Expected successful database destruction"); + DbLedger::destroy(&ledger_path).expect("Expected successful database destruction"); } #[test] @@ -612,8 +620,7 @@ mod tests { // Destroying database without closing it first is undefined behavior drop(ledger); - DB::destroy(&Options::default(), &ledger_path) - .expect("Expected successful database destruction"); + DbLedger::destroy(&ledger_path).expect("Expected successful database destruction"); } #[test] @@ -649,8 +656,7 @@ mod tests { // Destroying database without closing it first is undefined behavior drop(ledger); - DB::destroy(&Options::default(), &ledger_path) - .expect("Expected successful database destruction"); + DbLedger::destroy(&ledger_path).expect("Expected successful database destruction"); } #[test] @@ -689,7 +695,6 @@ mod tests { db_iterator.next(); } } - DB::destroy(&Options::default(), &db_ledger_path) - .expect("Expected successful database destruction"); + DbLedger::destroy(&db_ledger_path).expect("Expected successful database destruction"); } } diff --git a/src/db_window.rs b/src/db_window.rs index 1e08164d644948..413fe67f07882c 100644 --- a/src/db_window.rs +++ b/src/db_window.rs @@ -276,6 +276,7 @@ pub fn process_blob( // Check if the blob is in the range of our known leaders. If not, we return. // TODO: Need to update slot in broadcast, otherwise this check will fail with // leader rotation enabled + // Github issue: https://github.com/solana-labs/solana/issues/1899. let slot = blob.read().unwrap().slot()?; let leader = leader_scheduler.get_leader_for_slot(slot); diff --git a/src/fullnode.rs b/src/fullnode.rs index 752db1bf4ddcce..9f61681d3ca95d 100644 --- a/src/fullnode.rs +++ b/src/fullnode.rs @@ -3,7 +3,7 @@ use bank::Bank; use broadcast_stage::BroadcastStage; use cluster_info::{ClusterInfo, Node, NodeInfo}; -use db_ledger::DB_LEDGER_DIRECTORY; +use db_ledger::{write_entries_to_ledger, DbLedger}; use leader_scheduler::LeaderScheduler; use ledger::read_ledger; use ncp::Ncp; @@ -107,6 +107,7 @@ pub struct Fullnode { broadcast_socket: UdpSocket, rpc_addr: SocketAddr, rpc_pubsub_addr: SocketAddr, + db_ledger: Arc>, } #[derive(Serialize, Deserialize, Clone, Debug, PartialEq)] @@ -259,6 +260,10 @@ impl Fullnode { .expect("Leader not known after processing bank"); cluster_info.write().unwrap().set_leader(scheduled_leader); + + // Create the RocksDb ledger + let db_ledger = Self::make_db_ledger(ledger_path); + let node_role = if scheduled_leader != keypair.pubkey() { // Start in validator mode. let tvu = Tvu::new( @@ -282,7 +287,7 @@ impl Fullnode { .try_clone() .expect("Failed to clone retransmit socket"), Some(ledger_path), - &format!("{}/{}", ledger_path, DB_LEDGER_DIRECTORY), + db_ledger.clone(), ); let tpu_forwarder = TpuForwarder::new( node.sockets @@ -353,6 +358,7 @@ impl Fullnode { broadcast_socket: node.sockets.broadcast, rpc_addr, rpc_pubsub_addr, + db_ledger, } } @@ -435,7 +441,7 @@ impl Fullnode { .try_clone() .expect("Failed to clone retransmit socket"), Some(&self.ledger_path), - &format!("{}/{}", self.ledger_path, DB_LEDGER_DIRECTORY), + self.db_ledger.clone(), ); let tpu_forwarder = TpuForwarder::new( self.transaction_sockets @@ -590,6 +596,19 @@ impl Fullnode { ), ) } + + fn make_db_ledger(ledger_path: &str) -> Arc> { + // Destroy any existing instances of the RocksDb ledger + DbLedger::destroy(&ledger_path).expect("Expected successful database destruction"); + let ledger_entries = read_ledger(ledger_path, true) + .expect("opening ledger") + .map(|entry| entry.unwrap()); + + write_entries_to_ledger(&[ledger_path], ledger_entries); + let db = + DbLedger::open(ledger_path).expect("Expected to successfully open database ledger"); + Arc::new(RwLock::new(db)) + } } impl Service for Fullnode { @@ -630,9 +649,8 @@ mod tests { use db_ledger::*; use fullnode::{Fullnode, FullnodeReturnType, NodeRole, TvuReturnType}; use leader_scheduler::{make_active_set_entries, LeaderScheduler, LeaderSchedulerConfig}; - use ledger::{create_tmp_genesis, create_tmp_sample_ledger, LedgerWriter}; + use ledger::{create_tmp_genesis, create_tmp_sample_ledger, tmp_copy_ledger, LedgerWriter}; use packet::make_consecutive_blobs; - use rocksdb::{Options, DB}; use service::Service; use signature::{Keypair, KeypairUtil}; use std::cmp; @@ -842,6 +860,13 @@ mod tests { + num_ending_ticks as u64; ledger_writer.write_entries(&active_set_entries).unwrap(); + let validator_ledger_path = + tmp_copy_ledger(&bootstrap_leader_ledger_path, "test_wrong_role_transition"); + let ledger_paths = vec![ + bootstrap_leader_ledger_path.clone(), + validator_ledger_path.clone(), + ]; + // Create the common leader scheduling configuration let num_slots_per_epoch = 3; let leader_rotation_interval = 5; @@ -858,45 +883,53 @@ mod tests { Some(genesis_tick_height), ); - // Test that a node knows to transition to a validator based on parsing the ledger - let leader_vote_account_keypair = Arc::new(Keypair::new()); - let bootstrap_leader = Fullnode::new( - bootstrap_leader_node, - &bootstrap_leader_ledger_path, - bootstrap_leader_keypair, - leader_vote_account_keypair, - Some(bootstrap_leader_info.ncp), - false, - LeaderScheduler::new(&leader_scheduler_config), - None, - ); + { + // Test that a node knows to transition to a validator based on parsing the ledger + let leader_vote_account_keypair = Arc::new(Keypair::new()); + let bootstrap_leader = Fullnode::new( + bootstrap_leader_node, + &bootstrap_leader_ledger_path, + bootstrap_leader_keypair, + leader_vote_account_keypair, + Some(bootstrap_leader_info.ncp), + false, + LeaderScheduler::new(&leader_scheduler_config), + None, + ); - match bootstrap_leader.node_role { - Some(NodeRole::Validator(_)) => (), - _ => { - panic!("Expected bootstrap leader to be a validator"); + match bootstrap_leader.node_role { + Some(NodeRole::Validator(_)) => (), + _ => { + panic!("Expected bootstrap leader to be a validator"); + } } - } - // Test that a node knows to transition to a leader based on parsing the ledger - let validator = Fullnode::new( - validator_node, - &bootstrap_leader_ledger_path, - Arc::new(validator_keypair), - Arc::new(validator_vote_account_keypair), - Some(bootstrap_leader_info.ncp), - false, - LeaderScheduler::new(&leader_scheduler_config), - None, - ); + // Test that a node knows to transition to a leader based on parsing the ledger + let validator = Fullnode::new( + validator_node, + &validator_ledger_path, + Arc::new(validator_keypair), + Arc::new(validator_vote_account_keypair), + Some(bootstrap_leader_info.ncp), + false, + LeaderScheduler::new(&leader_scheduler_config), + None, + ); - match validator.node_role { - Some(NodeRole::Leader(_)) => (), - _ => { - panic!("Expected node to be the leader"); + match validator.node_role { + Some(NodeRole::Leader(_)) => (), + _ => { + panic!("Expected node to be the leader"); + } } + + validator.close().expect("Expected node to close"); + bootstrap_leader.close().expect("Expected node to close"); + } + for path in ledger_paths { + DbLedger::destroy(&path).expect("Expected successful database destruction"); + let _ignored = remove_dir_all(&path); } - let _ignored = remove_dir_all(&bootstrap_leader_ledger_path); } #[test] @@ -909,7 +942,7 @@ mod tests { // Create validator identity let num_ending_ticks = 1; - let (mint, validator_ledger_path, mut genesis_entries) = create_tmp_sample_ledger( + let (mint, validator_ledger_path, genesis_entries) = create_tmp_sample_ledger( "test_validator_to_leader_transition", 10_000, num_ending_ticks, @@ -946,11 +979,6 @@ mod tests { ledger_writer.write_entries(&active_set_entries).unwrap(); let ledger_initial_len = genesis_entries.len() as u64 + active_set_entries_len; - // Create RocksDb ledger, write genesis entries to it - let db_ledger_path = format!("{}/{}", validator_ledger_path, DB_LEDGER_DIRECTORY); - genesis_entries.extend(active_set_entries); - write_entries_to_ledger(&vec![db_ledger_path.clone()], &genesis_entries); - // Set the leader scheduler for the validator let leader_rotation_interval = 10; let num_bootstrap_slots = 2; @@ -1043,7 +1071,7 @@ mod tests { // Shut down t_responder.join().expect("responder thread join"); validator.close().unwrap(); - DB::destroy(&Options::default(), &db_ledger_path) + DbLedger::destroy(&validator_ledger_path) .expect("Expected successful database destruction"); let _ignored = remove_dir_all(&validator_ledger_path).unwrap(); } diff --git a/src/ledger.rs b/src/ledger.rs index e4f4f0e37a9da0..578fb62970541e 100644 --- a/src/ledger.rs +++ b/src/ledger.rs @@ -13,7 +13,7 @@ use rayon::prelude::*; use signature::{Keypair, KeypairUtil}; use solana_sdk::hash::{hash, Hash}; use solana_sdk::pubkey::Pubkey; -use std::fs::{create_dir_all, remove_dir_all, File, OpenOptions}; +use std::fs::{copy, create_dir_all, remove_dir_all, File, OpenOptions}; use std::io::prelude::*; use std::io::{self, BufReader, BufWriter, Seek, SeekFrom}; use std::mem::size_of; @@ -638,6 +638,22 @@ pub fn create_tmp_sample_ledger( (mint, path, genesis) } +pub fn tmp_copy_ledger(from: &str, name: &str) -> String { + let tostr = get_tmp_ledger_path(name); + + { + let to = Path::new(&tostr); + let from = Path::new(&from); + + create_dir_all(to).unwrap(); + + copy(from.join("data"), to.join("data")).unwrap(); + copy(from.join("index"), to.join("index")).unwrap(); + } + + tostr +} + pub fn make_tiny_test_entries(num: usize) -> Vec { let zero = Hash::default(); let one = hash(&zero.as_ref()); diff --git a/src/tvu.rs b/src/tvu.rs index 05ed46e11434e8..e5e69c37a0b8fb 100644 --- a/src/tvu.rs +++ b/src/tvu.rs @@ -66,14 +66,8 @@ impl Tvu { repair_socket: UdpSocket, retransmit_socket: UdpSocket, ledger_path: Option<&str>, - db_ledger_path: &str, + db_ledger: Arc>, ) -> Self { - // Eventually will be passed into LedgerWriteStage as well to replace the ledger, - // so wrap the object in a Arc - let db_ledger = Arc::new(RwLock::new( - DbLedger::open(db_ledger_path).expect("Expected to be able to open database ledger"), - )); - let exit = Arc::new(AtomicBool::new(false)); let repair_socket = Arc::new(repair_socket); @@ -173,6 +167,7 @@ pub mod tests { use bank::Bank; use bincode::serialize; use cluster_info::{ClusterInfo, Node}; + use db_ledger::DbLedger; use entry::Entry; use leader_scheduler::LeaderScheduler; use ledger::get_tmp_ledger_path; @@ -273,6 +268,8 @@ pub mod tests { let vote_account_keypair = Arc::new(Keypair::new()); let mut cur_hash = Hash::default(); let db_ledger_path = get_tmp_ledger_path("test_replicate"); + let db_ledger = + DbLedger::open(&db_ledger_path).expect("Expected to successfully open ledger"); let tvu = Tvu::new( Arc::new(target1_keypair), vote_account_keypair, @@ -284,7 +281,7 @@ pub mod tests { target1.sockets.repair, target1.sockets.retransmit, None, - &db_ledger_path.clone(), + Arc::new(RwLock::new(db_ledger)), ); let mut alice_ref_balance = starting_balance; diff --git a/tests/multinode.rs b/tests/multinode.rs index 9feaf7088e8757..9caf30e5467a2f 100644 --- a/tests/multinode.rs +++ b/tests/multinode.rs @@ -2,21 +2,19 @@ extern crate log; extern crate bincode; extern crate chrono; -extern crate rocksdb; extern crate serde_json; extern crate solana; extern crate solana_sdk; -use rocksdb::{Options, DB}; use solana::blob_fetch_stage::BlobFetchStage; use solana::cluster_info::{ClusterInfo, Node, NodeInfo}; use solana::contact_info::ContactInfo; -use solana::db_ledger::{write_entries_to_ledger, DB_LEDGER_DIRECTORY}; +use solana::db_ledger::DbLedger; use solana::entry::{reconstruct_entries_from_blobs, Entry}; use solana::fullnode::{Fullnode, FullnodeReturnType}; use solana::leader_scheduler::{make_active_set_entries, LeaderScheduler, LeaderSchedulerConfig}; use solana::ledger::{ - create_tmp_genesis, create_tmp_sample_ledger, get_tmp_ledger_path, read_ledger, LedgerWindow, + create_tmp_genesis, create_tmp_sample_ledger, read_ledger, tmp_copy_ledger, LedgerWindow, LedgerWriter, }; use solana::logger; @@ -36,9 +34,8 @@ use solana_sdk::pubkey::Pubkey; use solana_sdk::timing::{duration_as_ms, duration_as_s}; use std::collections::{HashSet, VecDeque}; use std::env; -use std::fs::{copy, create_dir_all, remove_dir_all}; +use std::fs::remove_dir_all; use std::net::UdpSocket; -use std::path::Path; use std::sync::atomic::{AtomicBool, Ordering}; use std::sync::{Arc, RwLock}; use std::thread::{sleep, Builder, JoinHandle}; @@ -113,22 +110,6 @@ fn converge(leader: &NodeInfo, num_nodes: usize) -> Vec { rv } -fn tmp_copy_ledger(from: &str, name: &str) -> String { - let tostr = get_tmp_ledger_path(name); - - { - let to = Path::new(&tostr); - let from = Path::new(&from); - - create_dir_all(to).unwrap(); - - copy(from.join("data"), to.join("data")).unwrap(); - copy(from.join("index"), to.join("index")).unwrap(); - } - - tostr -} - fn make_tiny_test_entries(start_hash: Hash, num: usize) -> Vec { let mut id = start_hash; let mut num_hashes = 0; @@ -962,7 +943,7 @@ fn test_leader_validator_basic() { // Make a common mint and a genesis entry for both leader + validator ledgers let num_ending_ticks = 1; - let (mint, leader_ledger_path, mut genesis_entries) = create_tmp_sample_ledger( + let (mint, leader_ledger_path, genesis_entries) = create_tmp_sample_ledger( "test_leader_validator_basic", 10_000, num_ending_ticks, @@ -989,20 +970,6 @@ fn test_leader_validator_basic() { make_active_set_entries(&validator_keypair, &mint.keypair(), &last_id, &last_id, 0); ledger_writer.write_entries(&active_set_entries).unwrap(); - // Create RocksDb ledgers, write genesis entries to them - let db_leader_ledger_path = format!("{}/{}", leader_ledger_path, DB_LEDGER_DIRECTORY); - let db_validator_ledger_path = format!("{}/{}", validator_ledger_path, DB_LEDGER_DIRECTORY); - - // Write the validator entries to the validator database, they - // will have repair the missing leader "active_set_entries" - write_entries_to_ledger(&vec![db_validator_ledger_path.clone()], &genesis_entries); - - // Next write the leader entries to the leader - genesis_entries.extend(active_set_entries); - write_entries_to_ledger(&vec![db_leader_ledger_path.clone()], &genesis_entries); - - let db_ledger_paths = vec![db_leader_ledger_path, db_validator_ledger_path]; - // Create the leader scheduler config let num_bootstrap_slots = 2; let bootstrap_height = num_bootstrap_slots * leader_rotation_interval; @@ -1103,11 +1070,8 @@ fn test_leader_validator_basic() { assert!(min_len >= bootstrap_height); - for path in db_ledger_paths { - DB::destroy(&Options::default(), &path).expect("Expected successful database destruction"); - } - for path in ledger_paths { + DbLedger::destroy(&path).expect("Expected successful database destruction"); remove_dir_all(path).unwrap(); } } @@ -1308,7 +1272,7 @@ fn test_full_leader_validator_network() { // Make a common mint and a genesis entry for both leader + validator's ledgers let num_ending_ticks = 1; - let (mint, bootstrap_leader_ledger_path, mut genesis_entries) = create_tmp_sample_ledger( + let (mint, bootstrap_leader_ledger_path, genesis_entries) = create_tmp_sample_ledger( "test_full_leader_validator_network", 10_000, num_ending_ticks, @@ -1355,21 +1319,8 @@ fn test_full_leader_validator_network() { .expect("expected at least one genesis entry") .id; ledger_writer.write_entries(&bootstrap_entries).unwrap(); - genesis_entries.extend(bootstrap_entries); } - // Create RocksDb ledger for bootstrap leader, write genesis entries to them - let db_bootstrap_leader_ledger_path = - format!("{}/{}", bootstrap_leader_ledger_path, DB_LEDGER_DIRECTORY); - - // Write the validator entries to the validator databases. - write_entries_to_ledger( - &vec![db_bootstrap_leader_ledger_path.clone()], - &genesis_entries, - ); - - let mut db_ledger_paths = vec![db_bootstrap_leader_ledger_path]; - // Create the common leader scheduling configuration let num_slots_per_epoch = (N + 1) as u64; let num_bootstrap_slots = 2; @@ -1401,15 +1352,8 @@ fn test_full_leader_validator_network() { &bootstrap_leader_ledger_path, "test_full_leader_validator_network", ); - ledger_paths.push(validator_ledger_path.clone()); - - // Create RocksDb ledgers, write genesis entries to them - let db_validator_ledger_path = format!("{}/{}", validator_ledger_path, DB_LEDGER_DIRECTORY); - db_ledger_paths.push(db_validator_ledger_path.clone()); - // Write the validator entries to the validator database, they - // will have repair the missing leader "active_set_entries" - write_entries_to_ledger(&vec![db_validator_ledger_path.clone()], &genesis_entries); + ledger_paths.push(validator_ledger_path.clone()); let validator_id = kp.pubkey(); let validator_node = Node::new_localhost_with_pubkey(validator_id); @@ -1550,11 +1494,8 @@ fn test_full_leader_validator_network() { assert!(shortest.unwrap() >= target_height); - for path in db_ledger_paths { - DB::destroy(&Options::default(), &path).expect("Expected successful database destruction"); - } - for path in ledger_paths { + DbLedger::destroy(&path).expect("Expected successful database destruction"); remove_dir_all(path).unwrap(); } }