| 
									
										
										
										
											2019-01-21 10:48:40 -08:00
										 |  |  | use bincode::serialize;
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  | use jsonrpc_core::futures::{
 | 
					
						
							|  |  |  |     future::{self, Future},
 | 
					
						
							|  |  |  |     stream::Stream,
 | 
					
						
							|  |  |  | };
 | 
					
						
							|  |  |  | use jsonrpc_core_client::transports::ws;
 | 
					
						
							| 
									
										
										
										
											2019-01-29 17:03:32 -08:00
										 |  |  | use log::*;
 | 
					
						
							| 
									
										
										
										
											2019-09-24 14:10:59 -06:00
										 |  |  | use reqwest::{self, header::CONTENT_TYPE};
 | 
					
						
							| 
									
										
										
										
											2019-01-21 10:48:40 -08:00
										 |  |  | use serde_json::{json, Value};
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  | use solana_client::{
 | 
					
						
							|  |  |  |     rpc_client::{get_rpc_request_str, RpcClient},
 | 
					
						
							| 
									
										
										
										
											2020-04-04 16:13:26 -07:00
										 |  |  |     rpc_response::{Response, RpcSignatureResult},
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  | };
 | 
					
						
							| 
									
										
										
										
											2020-04-09 18:05:56 -07:00
										 |  |  | use solana_core::contact_info::ContactInfo;
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  | use solana_core::{rpc_pubsub::gen_client::Client as PubsubClient, validator::TestValidator};
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  | use solana_sdk::{
 | 
					
						
							| 
									
										
										
										
											2020-04-04 16:13:26 -07:00
										 |  |  |     commitment_config::CommitmentConfig, hash::Hash, pubkey::Pubkey, signature::Signer,
 | 
					
						
							|  |  |  |     system_transaction, transaction::Transaction,
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  | };
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  | use std::{
 | 
					
						
							|  |  |  |     collections::HashSet,
 | 
					
						
							|  |  |  |     fs::remove_dir_all,
 | 
					
						
							|  |  |  |     net::UdpSocket,
 | 
					
						
							|  |  |  |     sync::mpsc::channel,
 | 
					
						
							|  |  |  |     thread::sleep,
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  |     time::{Duration, Instant},
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  | };
 | 
					
						
							|  |  |  | use tokio::runtime::Runtime;
 | 
					
						
							| 
									
										
										
										
											2019-01-21 10:48:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-09 18:05:56 -07:00
										 |  |  | macro_rules! json_req {
 | 
					
						
							|  |  |  |     ($method: expr, $params: expr) => {{
 | 
					
						
							|  |  |  |         json!({
 | 
					
						
							|  |  |  |            "jsonrpc": "2.0",
 | 
					
						
							|  |  |  |            "id": 1,
 | 
					
						
							|  |  |  |            "method": $method,
 | 
					
						
							|  |  |  |            "params": $params,
 | 
					
						
							|  |  |  |         })
 | 
					
						
							|  |  |  |     }}
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fn post_rpc(request: Value, data: &ContactInfo) -> Value {
 | 
					
						
							|  |  |  |     let client = reqwest::blocking::Client::new();
 | 
					
						
							|  |  |  |     let rpc_addr = data.rpc;
 | 
					
						
							|  |  |  |     let rpc_string = get_rpc_request_str(rpc_addr, false);
 | 
					
						
							|  |  |  |     let response = client
 | 
					
						
							|  |  |  |         .post(&rpc_string)
 | 
					
						
							|  |  |  |         .header(CONTENT_TYPE, "application/json")
 | 
					
						
							|  |  |  |         .body(request.to_string())
 | 
					
						
							|  |  |  |         .send()
 | 
					
						
							|  |  |  |         .unwrap();
 | 
					
						
							|  |  |  |     serde_json::from_str(&response.text().unwrap()).unwrap()
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-21 10:48:40 -08:00
										 |  |  | #[test]
 | 
					
						
							|  |  |  | fn test_rpc_send_tx() {
 | 
					
						
							| 
									
										
										
										
											2019-01-29 17:03:32 -08:00
										 |  |  |     solana_logger::setup();
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |     let TestValidator {
 | 
					
						
							|  |  |  |         server,
 | 
					
						
							|  |  |  |         leader_data,
 | 
					
						
							|  |  |  |         alice,
 | 
					
						
							|  |  |  |         ledger_path,
 | 
					
						
							|  |  |  |         ..
 | 
					
						
							|  |  |  |     } = TestValidator::run();
 | 
					
						
							| 
									
										
										
										
											2019-03-30 21:37:33 -06:00
										 |  |  |     let bob_pubkey = Pubkey::new_rand();
 | 
					
						
							| 
									
										
										
										
											2019-01-21 10:48:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-09 18:05:56 -07:00
										 |  |  |     let req = json_req!("getRecentBlockhash", json!([]));
 | 
					
						
							|  |  |  |     let json = post_rpc(req, &leader_data);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-15 00:25:45 -07:00
										 |  |  |     let blockhash: Hash = json["result"]["value"]["blockhash"]
 | 
					
						
							| 
									
										
										
										
											2019-11-12 14:49:41 -05:00
										 |  |  |         .as_str()
 | 
					
						
							|  |  |  |         .unwrap()
 | 
					
						
							|  |  |  |         .parse()
 | 
					
						
							|  |  |  |         .unwrap();
 | 
					
						
							| 
									
										
										
										
											2019-01-29 17:03:32 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-02 10:25:16 -08:00
										 |  |  |     info!("blockhash: {:?}", blockhash);
 | 
					
						
							| 
									
										
										
										
											2019-05-20 10:03:19 -07:00
										 |  |  |     let tx = system_transaction::transfer(&alice, &bob_pubkey, 20, blockhash);
 | 
					
						
							| 
									
										
										
										
											2020-01-21 22:16:07 -07:00
										 |  |  |     let serialized_encoded_tx = bs58::encode(serialize(&tx).unwrap()).into_string();
 | 
					
						
							| 
									
										
										
										
											2019-01-21 10:48:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-09 18:05:56 -07:00
										 |  |  |     let req = json_req!("sendTransaction", json!([serialized_encoded_tx]));
 | 
					
						
							|  |  |  |     let json: Value = post_rpc(req, &leader_data);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-21 10:48:40 -08:00
										 |  |  |     let signature = &json["result"];
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let mut confirmed_tx = false;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-09 18:05:56 -07:00
										 |  |  |     let request = json_req!("confirmTransaction", [signature]);
 | 
					
						
							| 
									
										
										
										
											2019-01-21 10:48:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-06 14:30:56 -07:00
										 |  |  |     for _ in 0..solana_sdk::clock::DEFAULT_TICKS_PER_SLOT {
 | 
					
						
							| 
									
										
										
										
											2020-04-09 18:05:56 -07:00
										 |  |  |         let json = post_rpc(request.clone(), &leader_data);
 | 
					
						
							| 
									
										
										
										
											2019-01-21 10:48:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-12 14:49:41 -05:00
										 |  |  |         if true == json["result"]["value"] {
 | 
					
						
							| 
									
										
										
										
											2019-01-21 10:48:40 -08:00
										 |  |  |             confirmed_tx = true;
 | 
					
						
							|  |  |  |             break;
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-29 17:03:32 -08:00
										 |  |  |         sleep(Duration::from_millis(500));
 | 
					
						
							| 
									
										
										
										
											2019-01-21 10:48:40 -08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert_eq!(confirmed_tx, true);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-03 16:44:06 -08:00
										 |  |  |     server.close().unwrap();
 | 
					
						
							| 
									
										
										
										
											2019-01-21 10:48:40 -08:00
										 |  |  |     remove_dir_all(ledger_path).unwrap();
 | 
					
						
							|  |  |  | }
 | 
					
						
							| 
									
										
										
										
											2019-11-14 11:41:26 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | #[test]
 | 
					
						
							|  |  |  | fn test_rpc_invalid_requests() {
 | 
					
						
							|  |  |  |     solana_logger::setup();
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |     let TestValidator {
 | 
					
						
							|  |  |  |         server,
 | 
					
						
							|  |  |  |         leader_data,
 | 
					
						
							|  |  |  |         ledger_path,
 | 
					
						
							|  |  |  |         ..
 | 
					
						
							|  |  |  |     } = TestValidator::run();
 | 
					
						
							| 
									
										
										
										
											2019-11-14 11:41:26 -05:00
										 |  |  |     let bob_pubkey = Pubkey::new_rand();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // test invalid get_balance request
 | 
					
						
							| 
									
										
										
										
											2020-04-09 18:05:56 -07:00
										 |  |  |     let req = json_req!("getBalance", json!(["invalid9999"]));
 | 
					
						
							|  |  |  |     let json = post_rpc(req, &leader_data);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-14 11:41:26 -05:00
										 |  |  |     let the_error = json["error"]["message"].as_str().unwrap();
 | 
					
						
							|  |  |  |     assert_eq!(the_error, "Invalid request");
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // test invalid get_account_info request
 | 
					
						
							| 
									
										
										
										
											2020-04-09 18:05:56 -07:00
										 |  |  |     let req = json_req!("getAccountInfo", json!(["invalid9999"]));
 | 
					
						
							|  |  |  |     let json = post_rpc(req, &leader_data);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-14 11:41:26 -05:00
										 |  |  |     let the_error = json["error"]["message"].as_str().unwrap();
 | 
					
						
							|  |  |  |     assert_eq!(the_error, "Invalid request");
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // test invalid get_account_info request
 | 
					
						
							| 
									
										
										
										
											2020-04-09 18:05:56 -07:00
										 |  |  |     let req = json_req!("getAccountInfo", json!([bob_pubkey.to_string()]));
 | 
					
						
							|  |  |  |     let json = post_rpc(req, &leader_data);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-14 11:41:26 -05:00
										 |  |  |     let the_value = &json["result"]["value"];
 | 
					
						
							|  |  |  |     assert!(the_value.is_null());
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     server.close().unwrap();
 | 
					
						
							|  |  |  |     remove_dir_all(ledger_path).unwrap();
 | 
					
						
							|  |  |  | }
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #[test]
 | 
					
						
							|  |  |  | fn test_rpc_subscriptions() {
 | 
					
						
							|  |  |  |     solana_logger::setup();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let TestValidator {
 | 
					
						
							|  |  |  |         server,
 | 
					
						
							|  |  |  |         leader_data,
 | 
					
						
							|  |  |  |         alice,
 | 
					
						
							|  |  |  |         ledger_path,
 | 
					
						
							|  |  |  |         genesis_hash,
 | 
					
						
							|  |  |  |         ..
 | 
					
						
							|  |  |  |     } = TestValidator::run();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let transactions_socket = UdpSocket::bind("0.0.0.0:0").unwrap();
 | 
					
						
							| 
									
										
										
										
											2020-03-29 01:58:51 +08:00
										 |  |  |     transactions_socket.connect(leader_data.tpu).unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Create transaction signatures to subscribe to
 | 
					
						
							|  |  |  |     let transactions: Vec<Transaction> = (0..1000)
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  |         .map(|_| system_transaction::transfer(&alice, &Pubkey::new_rand(), 1, genesis_hash))
 | 
					
						
							|  |  |  |         .collect();
 | 
					
						
							|  |  |  |     let mut signature_set: HashSet<String> = transactions
 | 
					
						
							|  |  |  |         .iter()
 | 
					
						
							|  |  |  |         .map(|tx| tx.signatures[0].to_string())
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |         .collect();
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-29 01:58:51 +08:00
										 |  |  |     // Track when subscriptions are ready
 | 
					
						
							|  |  |  |     let (ready_sender, ready_receiver) = channel::<()>();
 | 
					
						
							|  |  |  |     // Track when status notifications are received
 | 
					
						
							| 
									
										
										
										
											2020-04-04 16:13:26 -07:00
										 |  |  |     let (status_sender, status_receiver) = channel::<(String, Response<RpcSignatureResult>)>();
 | 
					
						
							| 
									
										
										
										
											2020-03-29 01:58:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |     // Create the pub sub runtime
 | 
					
						
							|  |  |  |     let mut rt = Runtime::new().unwrap();
 | 
					
						
							|  |  |  |     let rpc_pubsub_url = format!("ws://{}/", leader_data.rpc_pubsub);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  |     // Subscribe to all signatures
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |     rt.spawn({
 | 
					
						
							|  |  |  |         let connect = ws::try_connect::<PubsubClient>(&rpc_pubsub_url).unwrap();
 | 
					
						
							|  |  |  |         let signature_set = signature_set.clone();
 | 
					
						
							|  |  |  |         connect
 | 
					
						
							|  |  |  |             .and_then(move |client| {
 | 
					
						
							|  |  |  |                 for sig in signature_set {
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  |                     let status_sender = status_sender.clone();
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |                     tokio::spawn(
 | 
					
						
							|  |  |  |                         client
 | 
					
						
							|  |  |  |                             .signature_subscribe(sig.clone(), None)
 | 
					
						
							|  |  |  |                             .and_then(move |sig_stream| {
 | 
					
						
							|  |  |  |                                 sig_stream.for_each(move |result| {
 | 
					
						
							| 
									
										
										
										
											2020-03-29 01:58:51 +08:00
										 |  |  |                                     status_sender.send((sig.clone(), result)).unwrap();
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |                                     future::ok(())
 | 
					
						
							|  |  |  |                                 })
 | 
					
						
							|  |  |  |                             })
 | 
					
						
							|  |  |  |                             .map_err(|err| {
 | 
					
						
							|  |  |  |                                 eprintln!("sig sub err: {:#?}", err);
 | 
					
						
							|  |  |  |                             }),
 | 
					
						
							|  |  |  |                     );
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							| 
									
										
										
										
											2020-03-29 01:58:51 +08:00
										 |  |  |                 tokio::spawn(
 | 
					
						
							|  |  |  |                     client
 | 
					
						
							|  |  |  |                         .slot_subscribe()
 | 
					
						
							|  |  |  |                         .and_then(move |slot_stream| {
 | 
					
						
							|  |  |  |                             slot_stream.for_each(move |_| {
 | 
					
						
							|  |  |  |                                 ready_sender.send(()).unwrap();
 | 
					
						
							|  |  |  |                                 future::ok(())
 | 
					
						
							|  |  |  |                             })
 | 
					
						
							|  |  |  |                         })
 | 
					
						
							|  |  |  |                         .map_err(|err| {
 | 
					
						
							|  |  |  |                             eprintln!("slot sub err: {:#?}", err);
 | 
					
						
							|  |  |  |                         }),
 | 
					
						
							|  |  |  |                 );
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |                 future::ok(())
 | 
					
						
							|  |  |  |             })
 | 
					
						
							|  |  |  |             .map_err(|_| ())
 | 
					
						
							|  |  |  |     });
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  |     // Wait for signature subscriptions
 | 
					
						
							| 
									
										
										
										
											2020-03-29 01:58:51 +08:00
										 |  |  |     ready_receiver.recv_timeout(Duration::from_secs(2)).unwrap();
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     let rpc_client = RpcClient::new_socket(leader_data.rpc);
 | 
					
						
							| 
									
										
										
										
											2020-03-29 01:58:51 +08:00
										 |  |  |     let mut mint_balance = rpc_client
 | 
					
						
							|  |  |  |         .get_balance_with_commitment(&alice.pubkey(), CommitmentConfig::recent())
 | 
					
						
							|  |  |  |         .unwrap()
 | 
					
						
							|  |  |  |         .value;
 | 
					
						
							|  |  |  |     assert!(mint_balance >= transactions.len() as u64);
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Send all transactions to tpu socket for processing
 | 
					
						
							|  |  |  |     transactions.iter().for_each(|tx| {
 | 
					
						
							|  |  |  |         transactions_socket
 | 
					
						
							| 
									
										
										
										
											2020-03-29 01:58:51 +08:00
										 |  |  |             .send(&bincode::serialize(&tx).unwrap())
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  |             .unwrap();
 | 
					
						
							|  |  |  |     });
 | 
					
						
							| 
									
										
										
										
											2020-03-29 01:58:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Track mint balance to know when transactions have completed
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  |     let now = Instant::now();
 | 
					
						
							| 
									
										
										
										
											2020-03-29 01:58:51 +08:00
										 |  |  |     let expected_mint_balance = mint_balance - transactions.len() as u64;
 | 
					
						
							|  |  |  |     while mint_balance != expected_mint_balance && now.elapsed() < Duration::from_secs(5) {
 | 
					
						
							|  |  |  |         mint_balance = rpc_client
 | 
					
						
							|  |  |  |             .get_balance_with_commitment(&alice.pubkey(), CommitmentConfig::recent())
 | 
					
						
							|  |  |  |             .unwrap()
 | 
					
						
							|  |  |  |             .value;
 | 
					
						
							|  |  |  |         sleep(Duration::from_millis(100));
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |     // Wait for all signature subscriptions
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  |     let deadline = Instant::now() + Duration::from_secs(5);
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |     while !signature_set.is_empty() {
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  |         let timeout = deadline.saturating_duration_since(Instant::now());
 | 
					
						
							|  |  |  |         match status_receiver.recv_timeout(timeout) {
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |             Ok((sig, result)) => {
 | 
					
						
							| 
									
										
										
										
											2020-04-04 16:13:26 -07:00
										 |  |  |                 assert!(result.value.err.is_none());
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |                 assert!(signature_set.remove(&sig));
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             Err(_err) => {
 | 
					
						
							| 
									
										
										
										
											2020-03-29 01:58:51 +08:00
										 |  |  |                 assert!(
 | 
					
						
							|  |  |  |                     false,
 | 
					
						
							| 
									
										
										
										
											2020-03-24 08:00:34 +08:00
										 |  |  |                     "recv_timeout, {}/{} signatures remaining",
 | 
					
						
							|  |  |  |                     signature_set.len(),
 | 
					
						
							|  |  |  |                     transactions.len()
 | 
					
						
							|  |  |  |                 );
 | 
					
						
							| 
									
										
										
										
											2020-02-26 12:23:54 +08:00
										 |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     rt.shutdown_now().wait().unwrap();
 | 
					
						
							|  |  |  |     server.close().unwrap();
 | 
					
						
							|  |  |  |     remove_dir_all(ledger_path).unwrap();
 | 
					
						
							|  |  |  | }
 |