2018-03-05 22:33:44 +03:00
|
|
|
// Copyright 2018 The Grin Developers
|
2016-11-30 05:51:36 +03:00
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2017-11-01 02:32:33 +03:00
|
|
|
extern crate grin_api as api;
|
|
|
|
extern crate grin_chain as chain;
|
2016-11-30 05:51:36 +03:00
|
|
|
extern crate grin_core as core;
|
2018-08-24 16:45:14 +03:00
|
|
|
extern crate grin_keychain as keychain;
|
2016-11-30 05:51:36 +03:00
|
|
|
extern crate grin_p2p as p2p;
|
2018-04-24 11:18:24 +03:00
|
|
|
extern crate grin_servers as servers;
|
2017-10-22 10:11:45 +03:00
|
|
|
extern crate grin_util as util;
|
2017-11-01 02:32:33 +03:00
|
|
|
extern crate grin_wallet as wallet;
|
2018-10-14 12:01:34 +03:00
|
|
|
#[macro_use]
|
2018-10-21 23:30:56 +03:00
|
|
|
extern crate log;
|
2016-12-15 03:10:39 +03:00
|
|
|
|
2017-06-27 05:09:01 +03:00
|
|
|
mod framework;
|
|
|
|
|
2018-11-10 06:27:52 +03:00
|
|
|
use std::cmp;
|
2018-06-05 20:26:32 +03:00
|
|
|
use std::default::Default;
|
2018-11-10 06:27:52 +03:00
|
|
|
use std::process::exit;
|
2018-10-14 12:01:34 +03:00
|
|
|
use std::sync::atomic::AtomicBool;
|
2018-10-20 03:13:07 +03:00
|
|
|
use std::sync::Arc;
|
2018-06-14 15:16:14 +03:00
|
|
|
use std::{thread, time};
|
2018-10-20 03:13:07 +03:00
|
|
|
use util::Mutex;
|
2016-11-30 05:51:36 +03:00
|
|
|
|
2018-07-08 19:37:09 +03:00
|
|
|
use core::core::hash::Hashed;
|
2018-06-14 15:16:14 +03:00
|
|
|
use core::global::{self, ChainTypes};
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2018-08-24 16:45:14 +03:00
|
|
|
use wallet::controller;
|
|
|
|
use wallet::libtx::slate::Slate;
|
2018-10-14 12:01:34 +03:00
|
|
|
use wallet::libwallet::types::{WalletBackend, WalletInst};
|
2018-08-24 16:45:14 +03:00
|
|
|
use wallet::lmdb_wallet::LMDBBackend;
|
|
|
|
use wallet::HTTPWalletClient;
|
|
|
|
use wallet::WalletConfig;
|
|
|
|
|
|
|
|
use framework::{
|
2018-10-14 12:01:34 +03:00
|
|
|
config, stop_all_servers, LocalServerContainerConfig, LocalServerContainerPool,
|
2018-08-24 16:45:14 +03:00
|
|
|
LocalServerContainerPoolConfig,
|
|
|
|
};
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-06-27 05:09:01 +03:00
|
|
|
/// Testing the frameworks by starting a fresh server, creating a genesis
|
|
|
|
/// Block and mining into a wallet for a bit
|
|
|
|
#[test]
|
2017-08-09 19:40:23 +03:00
|
|
|
fn basic_genesis_mine() {
|
2018-01-10 22:36:27 +03:00
|
|
|
util::init_test_logger();
|
2017-11-16 00:49:15 +03:00
|
|
|
global::set_mining_mode(ChainTypes::AutomatedTesting);
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
let test_name_dir = "genesis_mine";
|
|
|
|
framework::clean_all_output(test_name_dir);
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
// Create a server pool
|
|
|
|
let mut pool_config = LocalServerContainerPoolConfig::default();
|
|
|
|
pool_config.base_name = String::from(test_name_dir);
|
2018-03-09 20:16:31 +03:00
|
|
|
pool_config.run_length_in_seconds = 10;
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
pool_config.base_api_port = 30000;
|
|
|
|
pool_config.base_p2p_port = 31000;
|
|
|
|
pool_config.base_wallet_port = 32000;
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
let mut pool = LocalServerContainerPool::new(pool_config);
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
// Create a server to add into the pool
|
|
|
|
let mut server_config = LocalServerContainerConfig::default();
|
|
|
|
server_config.start_miner = true;
|
2018-01-10 22:36:27 +03:00
|
|
|
server_config.start_wallet = false;
|
|
|
|
server_config.burn_mining_rewards = true;
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
pool.create_server(&mut server_config);
|
2018-08-17 03:31:51 +03:00
|
|
|
let servers = pool.run_all_servers();
|
|
|
|
stop_all_servers(servers);
|
2017-06-16 19:47:29 +03:00
|
|
|
}
|
|
|
|
|
2017-06-27 05:09:01 +03:00
|
|
|
/// Creates 5 servers, first being a seed and check that through peer address
|
|
|
|
/// messages they all end up connected.
|
|
|
|
#[test]
|
2017-08-09 19:40:23 +03:00
|
|
|
fn simulate_seeding() {
|
2018-02-02 05:03:12 +03:00
|
|
|
util::init_test_logger();
|
2017-11-16 00:49:15 +03:00
|
|
|
global::set_mining_mode(ChainTypes::AutomatedTesting);
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
let test_name_dir = "simulate_seeding";
|
|
|
|
framework::clean_all_output(test_name_dir);
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
// Create a server pool
|
|
|
|
let mut pool_config = LocalServerContainerPoolConfig::default();
|
2018-10-14 12:01:34 +03:00
|
|
|
pool_config.base_name = test_name_dir.to_string();
|
2017-08-09 19:40:23 +03:00
|
|
|
pool_config.run_length_in_seconds = 30;
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-12-18 16:17:11 +03:00
|
|
|
// have to use different ports because of tests being run in parallel
|
2017-08-09 19:40:23 +03:00
|
|
|
pool_config.base_api_port = 30020;
|
|
|
|
pool_config.base_p2p_port = 31020;
|
|
|
|
pool_config.base_wallet_port = 32020;
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
let mut pool = LocalServerContainerPool::new(pool_config);
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
// Create a first seed server to add into the pool
|
|
|
|
let mut server_config = LocalServerContainerConfig::default();
|
|
|
|
// server_config.start_miner = true;
|
2018-01-10 22:36:27 +03:00
|
|
|
server_config.start_wallet = false;
|
|
|
|
server_config.burn_mining_rewards = true;
|
2017-08-09 19:40:23 +03:00
|
|
|
server_config.is_seeding = true;
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
pool.create_server(&mut server_config);
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2018-08-30 15:58:49 +03:00
|
|
|
// wait the seed server fully start up before start remaining servers
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
// point next servers at first seed
|
|
|
|
server_config.is_seeding = false;
|
2018-10-14 12:01:34 +03:00
|
|
|
server_config.seed_addr = format!(
|
2017-08-09 19:40:23 +03:00
|
|
|
"{}:{}",
|
2018-03-04 03:19:54 +03:00
|
|
|
server_config.base_addr, server_config.p2p_server_port
|
2018-10-14 12:01:34 +03:00
|
|
|
);
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2017-09-26 20:58:56 +03:00
|
|
|
for _ in 0..4 {
|
|
|
|
pool.create_server(&mut server_config);
|
|
|
|
}
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2018-08-17 03:31:51 +03:00
|
|
|
let servers = pool.run_all_servers();
|
2018-08-30 15:58:49 +03:00
|
|
|
thread::sleep(time::Duration::from_secs(5));
|
|
|
|
|
|
|
|
// Check they all end up connected.
|
2018-09-02 00:25:47 +03:00
|
|
|
let url = format!(
|
|
|
|
"http://{}:{}/v1/peers/connected",
|
|
|
|
&server_config.base_addr, 30020
|
2018-08-30 15:58:49 +03:00
|
|
|
);
|
2018-10-09 10:27:34 +03:00
|
|
|
let peers_all = api::client::get::<Vec<p2p::types::PeerInfoDisplay>>(url.as_str(), None);
|
2018-09-02 00:25:47 +03:00
|
|
|
assert!(peers_all.is_ok());
|
|
|
|
assert_eq!(peers_all.unwrap().len(), 4);
|
2018-08-30 15:58:49 +03:00
|
|
|
|
2018-08-17 03:31:51 +03:00
|
|
|
stop_all_servers(servers);
|
2018-08-30 15:58:49 +03:00
|
|
|
|
|
|
|
// wait servers fully stop before start next automated test
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
2017-06-16 19:47:29 +03:00
|
|
|
}
|
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
/// Create 1 server, start it mining, then connect 4 other peers mining and
|
2018-10-14 12:01:34 +03:00
|
|
|
/// using the first as a seed. Meant to test the evolution of mining difficulty with miners
|
|
|
|
/// running at different rates.
|
|
|
|
///
|
|
|
|
/// TODO: Just going to comment this out as an automatically run test for the time
|
|
|
|
/// being, As it's more for actively testing and hurts CI a lot
|
|
|
|
#[ignore]
|
|
|
|
#[test]
|
2017-08-09 19:40:23 +03:00
|
|
|
fn simulate_parallel_mining() {
|
2017-11-16 00:49:15 +03:00
|
|
|
global::set_mining_mode(ChainTypes::AutomatedTesting);
|
2017-06-27 05:09:01 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
let test_name_dir = "simulate_parallel_mining";
|
|
|
|
// framework::clean_all_output(test_name_dir);
|
2017-06-27 05:09:01 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
// Create a server pool
|
|
|
|
let mut pool_config = LocalServerContainerPoolConfig::default();
|
2018-10-14 12:01:34 +03:00
|
|
|
pool_config.base_name = test_name_dir.to_string();
|
2017-08-09 19:40:23 +03:00
|
|
|
pool_config.run_length_in_seconds = 60;
|
2017-12-18 16:17:11 +03:00
|
|
|
// have to use different ports because of tests being run in parallel
|
2017-08-09 19:40:23 +03:00
|
|
|
pool_config.base_api_port = 30040;
|
|
|
|
pool_config.base_p2p_port = 31040;
|
|
|
|
pool_config.base_wallet_port = 32040;
|
2017-06-27 05:09:01 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
let mut pool = LocalServerContainerPool::new(pool_config);
|
2017-06-27 05:09:01 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
// Create a first seed server to add into the pool
|
|
|
|
let mut server_config = LocalServerContainerConfig::default();
|
|
|
|
server_config.start_miner = true;
|
|
|
|
server_config.start_wallet = true;
|
|
|
|
server_config.is_seeding = true;
|
2017-06-27 05:09:01 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
pool.create_server(&mut server_config);
|
2017-06-27 05:09:01 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
// point next servers at first seed
|
|
|
|
server_config.is_seeding = false;
|
2018-10-14 12:01:34 +03:00
|
|
|
server_config.seed_addr = format!(
|
2017-08-09 19:40:23 +03:00
|
|
|
"{}:{}",
|
2018-03-04 03:19:54 +03:00
|
|
|
server_config.base_addr, server_config.p2p_server_port
|
2018-10-14 12:01:34 +03:00
|
|
|
);
|
2017-06-27 05:09:01 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
// And create 4 more, then let them run for a while
|
|
|
|
for i in 1..4 {
|
|
|
|
// fudge in some slowdown
|
|
|
|
server_config.miner_slowdown_in_millis = i * 2;
|
|
|
|
pool.create_server(&mut server_config);
|
|
|
|
}
|
2017-06-16 19:47:29 +03:00
|
|
|
|
2018-02-02 05:03:12 +03:00
|
|
|
// pool.connect_all_peers();
|
2017-06-27 05:09:01 +03:00
|
|
|
|
2018-08-17 03:31:51 +03:00
|
|
|
let servers = pool.run_all_servers();
|
|
|
|
stop_all_servers(servers);
|
2017-06-27 05:09:01 +03:00
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
// Check mining difficulty here?, though I'd think it's more valuable
|
2018-03-04 03:19:54 +03:00
|
|
|
// to simply output it. Can at least see the evolution of the difficulty target
|
|
|
|
// in the debug log output for now
|
2017-06-16 19:47:29 +03:00
|
|
|
}
|
|
|
|
|
2017-08-09 19:40:23 +03:00
|
|
|
// TODO: Convert these tests to newer framework format
|
2017-02-19 05:42:34 +03:00
|
|
|
/// Create a network of 5 servers and mine a block, verifying that the block
|
|
|
|
/// gets propagated to all.
|
2016-11-30 05:51:36 +03:00
|
|
|
#[test]
|
2018-03-23 01:04:22 +03:00
|
|
|
fn simulate_block_propagation() {
|
2017-11-01 02:32:33 +03:00
|
|
|
util::init_test_logger();
|
2017-12-04 22:16:57 +03:00
|
|
|
|
|
|
|
// we actually set the chain_type in the ServerConfig below
|
|
|
|
// TODO - avoid needing to set it in two places?
|
2017-11-16 00:49:15 +03:00
|
|
|
global::set_mining_mode(ChainTypes::AutomatedTesting);
|
2017-08-09 19:40:23 +03:00
|
|
|
|
|
|
|
let test_name_dir = "grin-prop";
|
|
|
|
framework::clean_all_output(test_name_dir);
|
|
|
|
|
|
|
|
// instantiates 5 servers on different ports
|
|
|
|
let mut servers = vec![];
|
|
|
|
for n in 0..5 {
|
2018-04-13 16:42:25 +03:00
|
|
|
let s = servers::Server::new(framework::config(10 * n, test_name_dir, 0)).unwrap();
|
2017-08-09 19:40:23 +03:00
|
|
|
servers.push(s);
|
2018-03-23 01:04:22 +03:00
|
|
|
thread::sleep(time::Duration::from_millis(100));
|
2017-08-09 19:40:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// start mining
|
2018-10-14 12:01:34 +03:00
|
|
|
let stop = Arc::new(AtomicBool::new(false));
|
|
|
|
servers[0].start_test_miner(None, stop.clone());
|
2017-08-09 19:40:23 +03:00
|
|
|
|
|
|
|
// monitor for a change of head on a different server and check whether
|
2017-11-14 21:57:16 +03:00
|
|
|
// chain height has changed
|
2018-09-04 11:52:11 +03:00
|
|
|
let mut success = false;
|
|
|
|
let mut time_spent = 0;
|
2018-02-02 05:03:12 +03:00
|
|
|
loop {
|
|
|
|
let mut count = 0;
|
|
|
|
for n in 0..5 {
|
|
|
|
if servers[n].head().height > 3 {
|
|
|
|
count += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if count == 5 {
|
2018-09-04 11:52:11 +03:00
|
|
|
success = true;
|
2018-02-02 05:03:12 +03:00
|
|
|
break;
|
|
|
|
}
|
2018-07-08 19:37:09 +03:00
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
2018-09-04 11:52:11 +03:00
|
|
|
time_spent += 1;
|
2018-10-14 12:01:34 +03:00
|
|
|
if time_spent >= 30 {
|
2018-10-21 23:30:56 +03:00
|
|
|
info!("simulate_block_propagation - fail on timeout",);
|
2018-09-04 11:52:11 +03:00
|
|
|
break;
|
|
|
|
}
|
2018-10-14 12:01:34 +03:00
|
|
|
|
|
|
|
// stop mining after 8s
|
|
|
|
if time_spent == 8 {
|
|
|
|
servers[0].stop_test_miner(stop.clone());
|
|
|
|
}
|
2018-02-02 05:03:12 +03:00
|
|
|
}
|
2018-03-23 01:04:22 +03:00
|
|
|
for n in 0..5 {
|
|
|
|
servers[n].stop();
|
|
|
|
}
|
2018-09-04 11:52:11 +03:00
|
|
|
assert_eq!(true, success);
|
|
|
|
|
|
|
|
// wait servers fully stop before start next automated test
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
2016-12-23 23:15:36 +03:00
|
|
|
}
|
|
|
|
|
2017-02-19 05:42:34 +03:00
|
|
|
/// Creates 2 different disconnected servers, mine a few blocks on one, connect
|
|
|
|
/// them and check that the 2nd gets all the blocks
|
2017-02-08 00:52:17 +03:00
|
|
|
#[test]
|
|
|
|
fn simulate_full_sync() {
|
2017-10-22 10:11:45 +03:00
|
|
|
util::init_test_logger();
|
2017-12-04 22:16:57 +03:00
|
|
|
|
|
|
|
// we actually set the chain_type in the ServerConfig below
|
2017-11-16 00:49:15 +03:00
|
|
|
global::set_mining_mode(ChainTypes::AutomatedTesting);
|
2017-08-09 19:40:23 +03:00
|
|
|
|
|
|
|
let test_name_dir = "grin-sync";
|
|
|
|
framework::clean_all_output(test_name_dir);
|
|
|
|
|
2018-04-13 16:42:25 +03:00
|
|
|
let s1 = servers::Server::new(framework::config(1000, "grin-sync", 1000)).unwrap();
|
2017-08-09 19:40:23 +03:00
|
|
|
// mine a few blocks on server 1
|
2018-10-14 12:01:34 +03:00
|
|
|
let stop = Arc::new(AtomicBool::new(false));
|
|
|
|
s1.start_test_miner(None, stop.clone());
|
2018-02-10 01:32:16 +03:00
|
|
|
thread::sleep(time::Duration::from_secs(8));
|
2018-10-14 12:01:34 +03:00
|
|
|
s1.stop_test_miner(stop);
|
2017-08-09 19:40:23 +03:00
|
|
|
|
2018-07-08 19:37:09 +03:00
|
|
|
let s2 = servers::Server::new(framework::config(1001, "grin-sync", 1000)).unwrap();
|
|
|
|
|
|
|
|
// Get the current header from s1.
|
|
|
|
let s1_header = s1.chain.head_header().unwrap();
|
2018-10-14 12:01:34 +03:00
|
|
|
info!(
|
|
|
|
"simulate_full_sync - s1 header head: {} at {}",
|
|
|
|
s1_header.hash(),
|
|
|
|
s1_header.height
|
|
|
|
);
|
2018-07-08 19:37:09 +03:00
|
|
|
|
|
|
|
// Wait for s2 to sync up to and including the header from s1.
|
2018-09-04 11:52:11 +03:00
|
|
|
let mut time_spent = 0;
|
2018-07-08 19:37:09 +03:00
|
|
|
while s2.head().height < s1_header.height {
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
2018-09-04 11:52:11 +03:00
|
|
|
time_spent += 1;
|
2018-10-14 12:01:34 +03:00
|
|
|
if time_spent >= 30 {
|
|
|
|
info!(
|
2018-09-26 23:38:44 +03:00
|
|
|
"sync fail. s2.head().height: {}, s1_header.height: {}",
|
|
|
|
s2.head().height,
|
|
|
|
s1_header.height
|
|
|
|
);
|
2018-09-04 11:52:11 +03:00
|
|
|
break;
|
|
|
|
}
|
2017-08-09 19:40:23 +03:00
|
|
|
}
|
2018-07-08 19:37:09 +03:00
|
|
|
|
|
|
|
// Confirm both s1 and s2 see a consistent header at that height.
|
|
|
|
let s2_header = s2.chain.get_block_header(&s1_header.hash()).unwrap();
|
|
|
|
assert_eq!(s1_header, s2_header);
|
|
|
|
|
|
|
|
// Stop our servers cleanly.
|
2018-03-23 01:04:22 +03:00
|
|
|
s1.stop();
|
|
|
|
s2.stop();
|
2018-09-04 11:52:11 +03:00
|
|
|
|
|
|
|
// wait servers fully stop before start next automated test
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
2016-12-23 23:15:36 +03:00
|
|
|
}
|
|
|
|
|
2018-02-10 01:32:16 +03:00
|
|
|
/// Creates 2 different disconnected servers, mine a few blocks on one, connect
|
|
|
|
/// them and check that the 2nd gets all using fast sync algo
|
|
|
|
#[test]
|
|
|
|
fn simulate_fast_sync() {
|
|
|
|
util::init_test_logger();
|
|
|
|
|
|
|
|
// we actually set the chain_type in the ServerConfig below
|
|
|
|
global::set_mining_mode(ChainTypes::AutomatedTesting);
|
|
|
|
|
|
|
|
let test_name_dir = "grin-fast";
|
|
|
|
framework::clean_all_output(test_name_dir);
|
|
|
|
|
2018-07-08 19:37:09 +03:00
|
|
|
// start s1 and mine enough blocks to get beyond the fast sync horizon
|
2018-04-13 16:42:25 +03:00
|
|
|
let s1 = servers::Server::new(framework::config(2000, "grin-fast", 2000)).unwrap();
|
2018-10-14 12:01:34 +03:00
|
|
|
let stop = Arc::new(AtomicBool::new(false));
|
|
|
|
s1.start_test_miner(None, stop.clone());
|
2018-07-08 19:37:09 +03:00
|
|
|
|
2018-08-23 02:51:25 +03:00
|
|
|
while s1.head().height < 20 {
|
2018-07-08 19:37:09 +03:00
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
}
|
2018-10-14 12:01:34 +03:00
|
|
|
s1.stop_test_miner(stop);
|
2018-02-10 01:32:16 +03:00
|
|
|
|
2018-03-23 01:04:22 +03:00
|
|
|
let mut conf = config(2001, "grin-fast", 2000);
|
2018-02-13 21:08:21 +03:00
|
|
|
conf.archive_mode = Some(false);
|
2018-07-08 19:37:09 +03:00
|
|
|
|
2018-04-12 12:11:24 +03:00
|
|
|
let s2 = servers::Server::new(conf).unwrap();
|
2018-08-23 02:51:25 +03:00
|
|
|
|
2018-07-08 19:37:09 +03:00
|
|
|
// Get the current header from s1.
|
|
|
|
let s1_header = s1.chain.head_header().unwrap();
|
|
|
|
|
|
|
|
// Wait for s2 to sync up to and including the header from s1.
|
2018-10-14 12:01:34 +03:00
|
|
|
let mut total_wait = 0;
|
2018-07-08 19:37:09 +03:00
|
|
|
while s2.head().height < s1_header.height {
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
2018-10-14 12:01:34 +03:00
|
|
|
total_wait += 1;
|
|
|
|
if total_wait >= 30 {
|
|
|
|
error!(
|
|
|
|
"simulate_fast_sync test fail on timeout! s2 height: {}, s1 height: {}",
|
|
|
|
s2.head().height,
|
|
|
|
s1_header.height,
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
2018-07-08 19:37:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Confirm both s1 and s2 see a consistent header at that height.
|
|
|
|
let s2_header = s2.chain.get_block_header(&s1_header.hash()).unwrap();
|
|
|
|
assert_eq!(s1_header, s2_header);
|
|
|
|
|
|
|
|
// Stop our servers cleanly.
|
2018-03-23 01:04:22 +03:00
|
|
|
s1.stop();
|
|
|
|
s2.stop();
|
2018-09-04 11:52:11 +03:00
|
|
|
|
|
|
|
// wait servers fully stop before start next automated test
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
2018-02-10 01:32:16 +03:00
|
|
|
}
|
|
|
|
|
2018-11-10 06:27:52 +03:00
|
|
|
/// Preparation:
|
|
|
|
/// Creates 6 disconnected servers: A, B, C, D, E and F, mine 80 blocks on A,
|
|
|
|
/// Compact server A.
|
|
|
|
/// Connect all servers, check all get state_sync_threshold full blocks using fast sync.
|
|
|
|
/// Disconnect all servers from each other.
|
|
|
|
///
|
|
|
|
/// Test case 1: nodes that just synced is able to handle forks of up to state_sync_threshold
|
|
|
|
/// Mine state_sync_threshold-7 blocks on A
|
|
|
|
/// Mine state_sync_threshold-1 blocks on C (long fork), connect C to server A
|
|
|
|
/// check server A can sync to C without txhashset download.
|
|
|
|
///
|
|
|
|
/// Test case 2: nodes with history in between state_sync_threshold and cut_through_horizon will
|
|
|
|
/// be able to handle forks larger than state_sync_threshold but not as large as cut_through_horizon.
|
|
|
|
/// Mine 20 blocks on A (then A has 59 blocks in local chain)
|
|
|
|
/// Mine cut_through_horizon-1 blocks on D (longer fork), connect D to servers A, then fork point
|
|
|
|
/// is at A's body head.height - 39, and 20 < 39 < 70.
|
|
|
|
/// check server A can sync without txhashset download.
|
|
|
|
///
|
|
|
|
/// Test case 3: nodes that have enough history is able to handle forks of up to cut_through_horizon
|
|
|
|
/// Mine cut_through_horizon+10 blocks on E, connect E to servers A and B
|
|
|
|
/// check server A can sync to E without txhashset download.
|
|
|
|
/// check server B can sync to E but need txhashset download.
|
|
|
|
///
|
|
|
|
/// Test case 4: nodes which had a success state sync can have a new state sync if needed.
|
|
|
|
/// Mine cut_through_horizon+20 blocks on F (longer fork than E), connect F to servers B
|
|
|
|
/// check server B can sync to F with txhashset download.
|
|
|
|
///
|
|
|
|
/// Test case 5: normal sync (not a fork) should not trigger a txhashset download
|
|
|
|
/// Mine cut_through_horizon-10 blocks on F, connect F to servers B
|
|
|
|
/// check server B can sync to F without txhashset download.
|
|
|
|
///
|
|
|
|
/// Test case 6: far behind sync (not a fork) should trigger a txhashset download
|
|
|
|
/// Mine cut_through_horizon+1 blocks on F, connect F to servers B
|
|
|
|
/// check server B can sync to F with txhashset download.
|
|
|
|
///
|
|
|
|
///
|
2018-10-14 12:01:34 +03:00
|
|
|
#[ignore]
|
|
|
|
#[test]
|
2018-11-10 06:27:52 +03:00
|
|
|
fn simulate_long_fork() {
|
2018-02-13 03:38:52 +03:00
|
|
|
util::init_test_logger();
|
2018-11-10 06:27:52 +03:00
|
|
|
println!("starting simulate_long_fork");
|
2018-02-13 03:38:52 +03:00
|
|
|
|
|
|
|
// we actually set the chain_type in the ServerConfig below
|
|
|
|
global::set_mining_mode(ChainTypes::AutomatedTesting);
|
|
|
|
|
2018-11-10 06:27:52 +03:00
|
|
|
let test_name_dir = "grin-long-fork";
|
|
|
|
framework::clean_all_output(test_name_dir);
|
2018-02-13 03:38:52 +03:00
|
|
|
|
2018-11-10 06:27:52 +03:00
|
|
|
let s = long_fork_test_preparation();
|
|
|
|
for si in &s {
|
|
|
|
si.pause();
|
|
|
|
}
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
|
|
|
|
long_fork_test_case_1(&s);
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
|
|
|
|
long_fork_test_case_2(&s);
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
|
|
|
|
long_fork_test_case_3(&s);
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
|
|
|
|
long_fork_test_case_4(&s);
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
|
|
|
|
long_fork_test_case_5(&s);
|
2018-02-13 03:38:52 +03:00
|
|
|
|
2018-11-10 06:27:52 +03:00
|
|
|
// Clean up
|
|
|
|
for si in &s {
|
|
|
|
si.stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
// wait servers fully stop before start next automated test
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
}
|
|
|
|
|
|
|
|
fn long_fork_test_preparation() -> Vec<servers::Server> {
|
|
|
|
println!("preparation: mine 80 blocks, create 6 servers and sync all of them");
|
|
|
|
|
|
|
|
let mut s: Vec<servers::Server> = vec![];
|
|
|
|
|
|
|
|
// start server A and mine 80 blocks to get beyond the fast sync horizon
|
|
|
|
let mut conf = framework::config(2100, "grin-long-fork", 2100);
|
|
|
|
conf.archive_mode = Some(false);
|
|
|
|
conf.api_secret_path = None;
|
|
|
|
let s0 = servers::Server::new(conf).unwrap();
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
s.push(s0);
|
|
|
|
let stop = Arc::new(AtomicBool::new(false));
|
|
|
|
s[0].start_test_miner(None, stop.clone());
|
|
|
|
|
|
|
|
while s[0].head().height < global::cut_through_horizon() as u64 + 10 {
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
}
|
|
|
|
s[0].stop_test_miner(stop);
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
|
|
|
|
// Get the current header from s0.
|
|
|
|
let s0_header = s[0].chain.head().unwrap();
|
|
|
|
|
|
|
|
// check the tail after compacting
|
|
|
|
let _ = s[0].chain.compact();
|
|
|
|
let s0_tail = s[0].chain.tail().unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
s0_header.height - global::cut_through_horizon() as u64,
|
|
|
|
s0_tail.height
|
|
|
|
);
|
|
|
|
|
|
|
|
for i in 1..6 {
|
|
|
|
let mut conf = config(2100 + i, "grin-long-fork", 2100);
|
2018-02-13 21:08:21 +03:00
|
|
|
conf.archive_mode = Some(false);
|
2018-11-10 06:27:52 +03:00
|
|
|
conf.api_secret_path = None;
|
|
|
|
let si = servers::Server::new(conf).unwrap();
|
|
|
|
s.push(si);
|
|
|
|
}
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
|
|
|
|
// Wait for s[1..5] to sync up to and including the header from s0.
|
|
|
|
let mut total_wait = 0;
|
|
|
|
let mut min_height = 0;
|
|
|
|
while min_height < s0_header.height {
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
total_wait += 1;
|
|
|
|
if total_wait >= 60 {
|
|
|
|
println!(
|
|
|
|
"simulate_long_fork (preparation) test fail on timeout! minimum height: {}, s0 height: {}",
|
|
|
|
min_height,
|
|
|
|
s0_header.height,
|
|
|
|
);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
min_height = s0_header.height;
|
|
|
|
for i in 1..6 {
|
|
|
|
min_height = cmp::min(s[i].head().height, min_height);
|
2018-02-13 03:38:52 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-10 06:27:52 +03:00
|
|
|
// Confirm both s0 and s1 see a consistent header at that height.
|
|
|
|
let s1_header = s[1].chain.head().unwrap();
|
|
|
|
assert_eq!(s0_header, s1_header);
|
|
|
|
println!(
|
|
|
|
"preparation done. all 5 servers head.height: {}",
|
|
|
|
s0_header.height
|
|
|
|
);
|
|
|
|
|
|
|
|
// Wait for peers fully connection
|
|
|
|
let mut total_wait = 0;
|
|
|
|
let mut min_peers = 0;
|
|
|
|
while min_peers < 4 {
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
total_wait += 1;
|
|
|
|
if total_wait >= 60 {
|
|
|
|
println!(
|
|
|
|
"simulate_long_fork (preparation) test fail on timeout! minimum connected peers: {}",
|
|
|
|
min_peers,
|
|
|
|
);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
min_peers = 4;
|
|
|
|
for i in 0..5 {
|
|
|
|
let peers_connected = get_connected_peers(&"127.0.0.1".to_owned(), 22100 + i);
|
|
|
|
min_peers = cmp::min(min_peers, peers_connected.len());
|
|
|
|
}
|
2018-02-13 03:38:52 +03:00
|
|
|
}
|
2018-09-04 11:52:11 +03:00
|
|
|
|
2018-11-10 06:27:52 +03:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn long_fork_test_mining(blocks: u64, n: u16, s: &servers::Server) {
|
|
|
|
// Get the current header from node.
|
|
|
|
let sn_header = s.chain.head().unwrap();
|
|
|
|
|
|
|
|
// Mining
|
|
|
|
let stop = Arc::new(AtomicBool::new(false));
|
|
|
|
s.start_test_miner(None, stop.clone());
|
|
|
|
|
|
|
|
while s.head().height < sn_header.height + blocks {
|
|
|
|
thread::sleep(time::Duration::from_millis(1));
|
|
|
|
}
|
|
|
|
s.stop_test_miner(stop);
|
2018-09-04 11:52:11 +03:00
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
2018-11-10 06:27:52 +03:00
|
|
|
println!(
|
|
|
|
"{} blocks mined on s{}. s{}.height: {} (old height: {})",
|
|
|
|
s.head().height - sn_header.height,
|
|
|
|
n,
|
|
|
|
n,
|
|
|
|
s.head().height,
|
|
|
|
sn_header.height,
|
|
|
|
);
|
|
|
|
|
|
|
|
let _ = s.chain.compact();
|
|
|
|
let sn_header = s.chain.head().unwrap();
|
|
|
|
let sn_tail = s.chain.tail().unwrap();
|
|
|
|
println!(
|
|
|
|
"after compacting, s{}.head().height: {}, s{}.tail().height: {}",
|
|
|
|
n, sn_header.height, n, sn_tail.height,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn long_fork_test_case_1(s: &Vec<servers::Server>) {
|
|
|
|
println!("\ntest case 1 start");
|
|
|
|
|
|
|
|
// Mine state_sync_threshold-7 blocks on s0
|
|
|
|
long_fork_test_mining(global::state_sync_threshold() as u64 - 7, 0, &s[0]);
|
|
|
|
|
|
|
|
// Mine state_sync_threshold-1 blocks on s2 (long fork), a fork with more work than s0 chain
|
|
|
|
long_fork_test_mining(global::state_sync_threshold() as u64 - 1, 2, &s[2]);
|
|
|
|
|
|
|
|
let s2_header = s[2].chain.head().unwrap();
|
|
|
|
let s0_header = s[0].chain.head().unwrap();
|
|
|
|
let s0_tail = s[0].chain.tail().unwrap();
|
|
|
|
println!(
|
|
|
|
"test case 1: s0 start syncing with s2... s0.head().height: {}, s2.head().height: {}",
|
|
|
|
s0_header.height, s2_header.height,
|
|
|
|
);
|
|
|
|
s[0].resume();
|
|
|
|
s[2].resume();
|
|
|
|
|
|
|
|
// Check server s0 can sync to s2 without txhashset download.
|
|
|
|
let mut total_wait = 0;
|
|
|
|
while s[0].head().height < s2_header.height {
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
total_wait += 1;
|
|
|
|
if total_wait >= 120 {
|
|
|
|
println!(
|
|
|
|
"test case 1: test fail on timeout! s0 height: {}, s2 height: {}",
|
|
|
|
s[0].head().height,
|
|
|
|
s2_header.height,
|
|
|
|
);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let s0_tail_new = s[0].chain.tail().unwrap();
|
|
|
|
assert_eq!(s0_tail_new.height, s0_tail.height);
|
|
|
|
println!(
|
|
|
|
"test case 1: s0.head().height: {}, s2_header.height: {}",
|
|
|
|
s[0].head().height,
|
|
|
|
s2_header.height,
|
|
|
|
);
|
|
|
|
assert_eq!(s[0].head().last_block_h, s2_header.last_block_h);
|
|
|
|
|
|
|
|
s[0].pause();
|
|
|
|
s[2].stop();
|
|
|
|
println!("test case 1 passed")
|
|
|
|
}
|
|
|
|
|
|
|
|
fn long_fork_test_case_2(s: &Vec<servers::Server>) {
|
|
|
|
println!("\ntest case 2 start");
|
|
|
|
|
|
|
|
// Mine 20 blocks on s0
|
|
|
|
long_fork_test_mining(20, 0, &s[0]);
|
|
|
|
|
|
|
|
// Mine cut_through_horizon-1 blocks on s3 (longer fork)
|
|
|
|
long_fork_test_mining(global::cut_through_horizon() as u64 - 1, 3, &s[3]);
|
|
|
|
let s3_header = s[3].chain.head().unwrap();
|
|
|
|
let s0_header = s[0].chain.head().unwrap();
|
|
|
|
let s0_tail = s[0].chain.tail().unwrap();
|
|
|
|
println!(
|
|
|
|
"test case 2: s0 start syncing with s3. s0.head().height: {}, s3.head().height: {}",
|
|
|
|
s0_header.height, s3_header.height,
|
|
|
|
);
|
|
|
|
s[0].resume();
|
|
|
|
s[3].resume();
|
|
|
|
|
|
|
|
// Check server s0 can sync to s3 without txhashset download.
|
|
|
|
let mut total_wait = 0;
|
|
|
|
while s[0].head().height < s3_header.height {
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
total_wait += 1;
|
|
|
|
if total_wait >= 120 {
|
|
|
|
println!(
|
|
|
|
"test case 2: test fail on timeout! s0 height: {}, s3 height: {}",
|
|
|
|
s[0].head().height,
|
|
|
|
s3_header.height,
|
|
|
|
);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let s0_tail_new = s[0].chain.tail().unwrap();
|
|
|
|
assert_eq!(s0_tail_new.height, s0_tail.height);
|
|
|
|
assert_eq!(s[0].head().hash(), s3_header.hash());
|
|
|
|
|
|
|
|
let _ = s[0].chain.compact();
|
|
|
|
let s0_header = s[0].chain.head().unwrap();
|
|
|
|
let s0_tail = s[0].chain.tail().unwrap();
|
|
|
|
println!(
|
|
|
|
"test case 2: after compacting, s0.head().height: {}, s0.tail().height: {}",
|
|
|
|
s0_header.height, s0_tail.height,
|
|
|
|
);
|
|
|
|
|
|
|
|
s[0].pause();
|
|
|
|
s[3].stop();
|
|
|
|
println!("test case 2 passed")
|
|
|
|
}
|
|
|
|
|
|
|
|
fn long_fork_test_case_3(s: &Vec<servers::Server>) {
|
|
|
|
println!("\ntest case 3 start");
|
|
|
|
|
|
|
|
// Mine cut_through_horizon+1 blocks on s4
|
|
|
|
long_fork_test_mining(global::cut_through_horizon() as u64 + 10, 4, &s[4]);
|
|
|
|
|
|
|
|
let s4_header = s[4].chain.head().unwrap();
|
|
|
|
let s0_header = s[0].chain.head().unwrap();
|
|
|
|
let s0_tail = s[0].chain.tail().unwrap();
|
|
|
|
let s1_header = s[1].chain.head().unwrap();
|
|
|
|
let s1_tail = s[1].chain.tail().unwrap();
|
|
|
|
println!(
|
|
|
|
"test case 3: s0/1 start syncing with s4. s0.head().height: {}, s0.tail().height: {}, s1.head().height: {}, s1.tail().height: {}, s4.head().height: {}",
|
|
|
|
s0_header.height, s0_tail.height,
|
|
|
|
s1_header.height, s1_tail.height,
|
|
|
|
s4_header.height,
|
|
|
|
);
|
|
|
|
s[0].resume();
|
|
|
|
s[4].resume();
|
|
|
|
|
|
|
|
// Check server s0 can sync to s4.
|
|
|
|
let mut total_wait = 0;
|
|
|
|
while s[0].head().height < s4_header.height {
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
total_wait += 1;
|
|
|
|
if total_wait >= 120 {
|
|
|
|
println!(
|
|
|
|
"test case 3: test fail on timeout! s0 height: {}, s4 height: {}",
|
|
|
|
s[0].head().height,
|
|
|
|
s4_header.height,
|
|
|
|
);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert_eq!(s[0].head().hash(), s4_header.hash());
|
|
|
|
|
|
|
|
s[0].stop();
|
|
|
|
s[1].resume();
|
|
|
|
|
|
|
|
// Check server s1 can sync to s4 but with txhashset download.
|
|
|
|
let mut total_wait = 0;
|
|
|
|
while s[1].head().height < s4_header.height {
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
total_wait += 1;
|
|
|
|
if total_wait >= 120 {
|
|
|
|
println!(
|
|
|
|
"test case 3: test fail on timeout! s1 height: {}, s4 height: {}",
|
|
|
|
s[1].head().height,
|
|
|
|
s4_header.height,
|
|
|
|
);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let s1_tail_new = s[1].chain.tail().unwrap();
|
|
|
|
println!(
|
|
|
|
"test case 3: s[1].tail().height: {}, old height: {}",
|
|
|
|
s1_tail_new.height, s1_tail.height
|
|
|
|
);
|
|
|
|
assert_ne!(s1_tail_new.height, s1_tail.height);
|
|
|
|
assert_eq!(s[1].head().hash(), s4_header.hash());
|
|
|
|
|
|
|
|
s[1].pause();
|
|
|
|
s[4].pause();
|
|
|
|
println!("test case 3 passed")
|
|
|
|
}
|
|
|
|
|
|
|
|
fn long_fork_test_case_4(s: &Vec<servers::Server>) {
|
|
|
|
println!("\ntest case 4 start");
|
|
|
|
|
|
|
|
let _ = s[1].chain.compact();
|
|
|
|
|
|
|
|
// Mine cut_through_horizon+20 blocks on s5 (longer fork than s4)
|
|
|
|
long_fork_test_mining(global::cut_through_horizon() as u64 + 20, 5, &s[5]);
|
|
|
|
|
|
|
|
let s5_header = s[5].chain.head().unwrap();
|
|
|
|
let s1_header = s[1].chain.head().unwrap();
|
|
|
|
let s1_tail = s[1].chain.tail().unwrap();
|
|
|
|
println!(
|
|
|
|
"test case 4: s1 start syncing with s5. s1.head().height: {}, s1.tail().height: {}, s5.head().height: {}",
|
|
|
|
s1_header.height, s1_tail.height,
|
|
|
|
s5_header.height,
|
|
|
|
);
|
|
|
|
s[1].resume();
|
|
|
|
s[5].resume();
|
|
|
|
|
|
|
|
// Check server s1 can sync to s5 with a new txhashset download.
|
|
|
|
let mut total_wait = 0;
|
|
|
|
while s[1].head().height < s5_header.height {
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
total_wait += 1;
|
|
|
|
if total_wait >= 120 {
|
|
|
|
println!(
|
|
|
|
"test case 4: test fail on timeout! s1 height: {}, s5 height: {}",
|
|
|
|
s[1].head().height,
|
|
|
|
s5_header.height,
|
|
|
|
);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let s1_tail_new = s[1].chain.tail().unwrap();
|
|
|
|
println!(
|
|
|
|
"test case 4: s[1].tail().height: {}, old height: {}",
|
|
|
|
s1_tail_new.height, s1_tail.height
|
|
|
|
);
|
|
|
|
assert_ne!(s1_tail_new.height, s1_tail.height);
|
|
|
|
assert_eq!(s[1].head().hash(), s5_header.hash());
|
|
|
|
|
|
|
|
s[1].pause();
|
|
|
|
s[5].pause();
|
|
|
|
|
|
|
|
println!("test case 4 passed")
|
|
|
|
}
|
|
|
|
|
|
|
|
fn long_fork_test_case_5(s: &Vec<servers::Server>) {
|
|
|
|
println!("\ntest case 5 start");
|
|
|
|
|
|
|
|
let _ = s[1].chain.compact();
|
|
|
|
|
|
|
|
// Mine cut_through_horizon-10 blocks on s5
|
|
|
|
long_fork_test_mining(global::cut_through_horizon() as u64 - 10, 5, &s[5]);
|
|
|
|
|
|
|
|
let s5_header = s[5].chain.head().unwrap();
|
|
|
|
let s1_header = s[1].chain.head().unwrap();
|
|
|
|
let s1_tail = s[1].chain.tail().unwrap();
|
|
|
|
println!(
|
|
|
|
"test case 5: s1 start syncing with s5. s1.head().height: {}, s1.tail().height: {}, s5.head().height: {}",
|
|
|
|
s1_header.height, s1_tail.height,
|
|
|
|
s5_header.height,
|
|
|
|
);
|
|
|
|
s[1].resume();
|
|
|
|
s[5].resume();
|
|
|
|
|
|
|
|
// Check server s1 can sync to s5 without a txhashset download (normal body sync)
|
|
|
|
let mut total_wait = 0;
|
|
|
|
while s[1].head().height < s5_header.height {
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
total_wait += 1;
|
|
|
|
if total_wait >= 120 {
|
|
|
|
println!(
|
|
|
|
"test case 5: test fail on timeout! s1 height: {}, s5 height: {}",
|
|
|
|
s[1].head().height,
|
|
|
|
s5_header.height,
|
|
|
|
);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let s1_tail_new = s[1].chain.tail().unwrap();
|
|
|
|
println!(
|
|
|
|
"test case 5: s[1].tail().height: {}, old height: {}",
|
|
|
|
s1_tail_new.height, s1_tail.height
|
|
|
|
);
|
|
|
|
assert_eq!(s1_tail_new.height, s1_tail.height);
|
|
|
|
assert_eq!(s[1].head().hash(), s5_header.hash());
|
|
|
|
|
|
|
|
s[1].pause();
|
|
|
|
s[5].pause();
|
|
|
|
|
|
|
|
println!("test case 5 passed")
|
|
|
|
}
|
|
|
|
|
|
|
|
fn long_fork_test_case_6(s: &Vec<servers::Server>) {
|
|
|
|
println!("\ntest case 6 start");
|
|
|
|
|
|
|
|
let _ = s[1].chain.compact();
|
|
|
|
|
|
|
|
// Mine cut_through_horizon+1 blocks on s5
|
|
|
|
long_fork_test_mining(global::cut_through_horizon() as u64 + 1, 5, &s[5]);
|
|
|
|
|
|
|
|
let s5_header = s[5].chain.head().unwrap();
|
|
|
|
let s1_header = s[1].chain.head().unwrap();
|
|
|
|
let s1_tail = s[1].chain.tail().unwrap();
|
|
|
|
println!(
|
|
|
|
"test case 6: s1 start syncing with s5. s1.head().height: {}, s1.tail().height: {}, s5.head().height: {}",
|
|
|
|
s1_header.height, s1_tail.height,
|
|
|
|
s5_header.height,
|
|
|
|
);
|
|
|
|
s[1].resume();
|
|
|
|
s[5].resume();
|
|
|
|
|
|
|
|
// Check server s1 can sync to s5 without a txhashset download (normal body sync)
|
|
|
|
let mut total_wait = 0;
|
|
|
|
while s[1].head().height < s5_header.height {
|
|
|
|
thread::sleep(time::Duration::from_millis(1_000));
|
|
|
|
total_wait += 1;
|
|
|
|
if total_wait >= 120 {
|
|
|
|
println!(
|
|
|
|
"test case 6: test fail on timeout! s1 height: {}, s5 height: {}",
|
|
|
|
s[1].head().height,
|
|
|
|
s5_header.height,
|
|
|
|
);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let s1_tail_new = s[1].chain.tail().unwrap();
|
|
|
|
println!(
|
|
|
|
"test case 6: s[1].tail().height: {}, old height: {}",
|
|
|
|
s1_tail_new.height, s1_tail.height
|
|
|
|
);
|
|
|
|
assert_eq!(s1_tail_new.height, s1_tail.height);
|
|
|
|
assert_eq!(s[1].head().hash(), s5_header.hash());
|
|
|
|
|
|
|
|
s[1].pause();
|
|
|
|
s[5].pause();
|
|
|
|
|
|
|
|
println!("test case 6 passed")
|
2018-02-13 03:38:52 +03:00
|
|
|
}
|
2018-08-24 16:45:14 +03:00
|
|
|
|
|
|
|
pub fn create_wallet(
|
|
|
|
dir: &str,
|
|
|
|
client: HTTPWalletClient,
|
2018-11-11 14:56:42 +03:00
|
|
|
) -> Arc<Mutex<WalletInst<HTTPWalletClient, keychain::ExtKeychain>>> {
|
2018-08-24 16:45:14 +03:00
|
|
|
let mut wallet_config = WalletConfig::default();
|
|
|
|
wallet_config.data_file_dir = String::from(dir);
|
|
|
|
let _ = wallet::WalletSeed::init_file(&wallet_config);
|
|
|
|
let mut wallet: LMDBBackend<HTTPWalletClient, keychain::ExtKeychain> =
|
|
|
|
LMDBBackend::new(wallet_config.clone(), "", client).unwrap_or_else(|e| {
|
|
|
|
panic!("Error creating wallet: {:?} Config: {:?}", e, wallet_config)
|
|
|
|
});
|
|
|
|
wallet.open_with_credentials().unwrap_or_else(|e| {
|
|
|
|
panic!(
|
|
|
|
"Error initializing wallet: {:?} Config: {:?}",
|
|
|
|
e, wallet_config
|
|
|
|
)
|
|
|
|
});
|
2018-11-11 14:56:42 +03:00
|
|
|
Arc::new(Mutex::new(wallet))
|
2018-08-24 16:45:14 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Intended to replicate https://github.com/mimblewimble/grin/issues/1325
|
|
|
|
#[ignore]
|
|
|
|
#[test]
|
|
|
|
fn replicate_tx_fluff_failure() {
|
|
|
|
util::init_test_logger();
|
|
|
|
global::set_mining_mode(ChainTypes::UserTesting);
|
|
|
|
framework::clean_all_output("tx_fluff");
|
|
|
|
|
|
|
|
// Create Wallet 1 (Mining Input) and start it listening
|
|
|
|
// Wallet 1 post to another node, just for fun
|
2018-09-27 22:45:48 +03:00
|
|
|
let client1 = HTTPWalletClient::new("http://127.0.0.1:23003", None);
|
2018-08-24 16:45:14 +03:00
|
|
|
let wallet1 = create_wallet("target/tmp/tx_fluff/wallet1", client1.clone());
|
|
|
|
let wallet1_handle = thread::spawn(move || {
|
2018-10-02 10:49:36 +03:00
|
|
|
controller::foreign_listener(wallet1, "127.0.0.1:33000", None)
|
2018-08-24 16:45:14 +03:00
|
|
|
.unwrap_or_else(|e| panic!("Error creating wallet1 listener: {:?}", e,));
|
|
|
|
});
|
|
|
|
|
|
|
|
// Create Wallet 2 (Recipient) and launch
|
2018-09-27 22:45:48 +03:00
|
|
|
let client2 = HTTPWalletClient::new("http://127.0.0.1:23001", None);
|
2018-08-24 16:45:14 +03:00
|
|
|
let wallet2 = create_wallet("target/tmp/tx_fluff/wallet2", client2.clone());
|
|
|
|
let wallet2_handle = thread::spawn(move || {
|
2018-10-02 10:49:36 +03:00
|
|
|
controller::foreign_listener(wallet2, "127.0.0.1:33001", None)
|
2018-08-24 16:45:14 +03:00
|
|
|
.unwrap_or_else(|e| panic!("Error creating wallet2 listener: {:?}", e,));
|
|
|
|
});
|
|
|
|
|
|
|
|
// Server 1 (mines into wallet 1)
|
|
|
|
let mut s1_config = framework::config(3000, "tx_fluff", 3000);
|
|
|
|
s1_config.test_miner_wallet_url = Some("http://127.0.0.1:33000".to_owned());
|
|
|
|
s1_config.dandelion_config.embargo_secs = Some(10);
|
|
|
|
s1_config.dandelion_config.patience_secs = Some(1);
|
|
|
|
s1_config.dandelion_config.relay_secs = Some(1);
|
|
|
|
let s1 = servers::Server::new(s1_config.clone()).unwrap();
|
|
|
|
// Mine off of server 1
|
2018-10-14 12:01:34 +03:00
|
|
|
s1.start_test_miner(s1_config.test_miner_wallet_url, s1.stop.clone());
|
2018-08-24 16:45:14 +03:00
|
|
|
thread::sleep(time::Duration::from_secs(5));
|
|
|
|
|
|
|
|
// Server 2 (another node)
|
|
|
|
let mut s2_config = framework::config(3001, "tx_fluff", 3001);
|
|
|
|
s2_config.p2p_config.seeds = Some(vec!["127.0.0.1:13000".to_owned()]);
|
|
|
|
s2_config.dandelion_config.embargo_secs = Some(10);
|
|
|
|
s2_config.dandelion_config.patience_secs = Some(1);
|
|
|
|
s2_config.dandelion_config.relay_secs = Some(1);
|
2018-10-14 12:01:34 +03:00
|
|
|
let _s2 = servers::Server::new(s2_config.clone()).unwrap();
|
2018-08-24 16:45:14 +03:00
|
|
|
|
|
|
|
let dl_nodes = 5;
|
|
|
|
|
|
|
|
for i in 0..dl_nodes {
|
|
|
|
// (create some stem nodes)
|
|
|
|
let mut s_config = framework::config(3002 + i, "tx_fluff", 3002 + i);
|
|
|
|
s_config.p2p_config.seeds = Some(vec!["127.0.0.1:13000".to_owned()]);
|
|
|
|
s_config.dandelion_config.embargo_secs = Some(10);
|
|
|
|
s_config.dandelion_config.patience_secs = Some(1);
|
|
|
|
s_config.dandelion_config.relay_secs = Some(1);
|
|
|
|
let _ = servers::Server::new(s_config.clone()).unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
thread::sleep(time::Duration::from_secs(10));
|
|
|
|
|
|
|
|
// get another instance of wallet1 (to update contents and perform a send)
|
|
|
|
let wallet1 = create_wallet("target/tmp/tx_fluff/wallet1", client1.clone());
|
|
|
|
|
|
|
|
let amount = 30_000_000_000;
|
|
|
|
let mut slate = Slate::blank(1);
|
|
|
|
|
2018-11-11 14:56:42 +03:00
|
|
|
wallet::controller::owner_single_use(wallet1, |api| {
|
2018-09-26 23:38:44 +03:00
|
|
|
slate = api
|
|
|
|
.issue_send_tx(
|
2018-08-24 16:45:14 +03:00
|
|
|
amount, // amount
|
|
|
|
2, // minimum confirmations
|
|
|
|
"http://127.0.0.1:33001", // dest
|
|
|
|
500, // max outputs
|
2018-08-24 17:51:32 +03:00
|
|
|
1000, // num change outputs
|
2018-08-24 16:45:14 +03:00
|
|
|
true, // select all outputs
|
|
|
|
).unwrap();
|
2018-08-24 17:51:13 +03:00
|
|
|
api.post_tx(&slate, false).unwrap();
|
2018-08-24 16:45:14 +03:00
|
|
|
Ok(())
|
|
|
|
}).unwrap();
|
|
|
|
|
|
|
|
// Give some time for propagation and mining
|
2018-08-24 17:51:13 +03:00
|
|
|
thread::sleep(time::Duration::from_secs(200));
|
2018-08-24 16:45:14 +03:00
|
|
|
|
|
|
|
// get another instance of wallet (to check contents)
|
|
|
|
let wallet2 = create_wallet("target/tmp/tx_fluff/wallet2", client2.clone());
|
|
|
|
|
2018-11-11 14:56:42 +03:00
|
|
|
wallet::controller::owner_single_use(wallet2, |api| {
|
2018-08-24 16:45:14 +03:00
|
|
|
let res = api.retrieve_summary_info(true).unwrap();
|
|
|
|
assert_eq!(res.1.amount_currently_spendable, amount);
|
|
|
|
Ok(())
|
|
|
|
}).unwrap();
|
|
|
|
}
|
2018-11-10 06:27:52 +03:00
|
|
|
|
|
|
|
fn get_connected_peers(
|
|
|
|
base_addr: &String,
|
|
|
|
api_server_port: u16,
|
|
|
|
) -> Vec<p2p::types::PeerInfoDisplay> {
|
|
|
|
let url = format!(
|
|
|
|
"http://{}:{}/v1/peers/connected",
|
|
|
|
base_addr, api_server_port
|
|
|
|
);
|
|
|
|
api::client::get::<Vec<p2p::types::PeerInfoDisplay>>(url.as_str(), None).unwrap()
|
|
|
|
}
|