2018-06-04 10:19:50 +02:00
|
|
|
// Copyright 2015-2018 Parity Technologies (UK) Ltd.
|
2016-12-06 19:23:15 +01:00
|
|
|
// This file is part of Parity.
|
|
|
|
|
|
|
|
// Parity is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
|
|
|
// Parity is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2017-07-29 21:56:42 +02:00
|
|
|
use std::sync::Arc;
|
2017-08-31 11:35:41 +02:00
|
|
|
use hash::keccak;
|
2018-01-10 13:35:18 +01:00
|
|
|
use ethereum_types::{U256, Address};
|
2018-04-09 16:14:33 +02:00
|
|
|
use io::{IoHandler, IoChannel};
|
|
|
|
use ethcore::client::{ChainInfo, ClientIoMessage};
|
2016-12-06 19:23:15 +01:00
|
|
|
use ethcore::spec::Spec;
|
|
|
|
use ethcore::miner::MinerService;
|
|
|
|
use ethcore::account_provider::AccountProvider;
|
2017-01-11 12:16:47 +01:00
|
|
|
use ethkey::{KeyPair, Secret};
|
2018-01-11 17:49:10 +01:00
|
|
|
use transaction::{Action, PendingTransaction, Transaction};
|
2016-12-06 19:23:15 +01:00
|
|
|
use super::helpers::*;
|
2018-01-10 13:35:18 +01:00
|
|
|
use SyncConfig;
|
2016-12-06 19:23:15 +01:00
|
|
|
|
2017-08-21 13:46:58 +02:00
|
|
|
fn new_tx(secret: &Secret, nonce: U256, chain_id: u64) -> PendingTransaction {
|
2016-12-15 19:10:33 +01:00
|
|
|
let signed = Transaction {
|
2016-12-12 17:24:48 +01:00
|
|
|
nonce: nonce.into(),
|
|
|
|
gas_price: 0.into(),
|
|
|
|
gas: 21000.into(),
|
|
|
|
action: Action::Call(Address::default()),
|
|
|
|
value: 0.into(),
|
|
|
|
data: Vec::new(),
|
2017-08-21 13:46:58 +02:00
|
|
|
}.sign(secret, Some(chain_id));
|
2016-12-15 19:10:33 +01:00
|
|
|
PendingTransaction::new(signed, None)
|
2016-12-12 17:24:48 +01:00
|
|
|
}
|
|
|
|
|
2016-12-06 19:23:15 +01:00
|
|
|
#[test]
|
2016-12-11 12:32:01 +01:00
|
|
|
fn authority_round() {
|
2017-08-31 11:35:41 +02:00
|
|
|
let s0 = KeyPair::from_secret_slice(&keccak("1")).unwrap();
|
|
|
|
let s1 = KeyPair::from_secret_slice(&keccak("0")).unwrap();
|
2017-01-18 18:49:50 +01:00
|
|
|
let ap = Arc::new(AccountProvider::transient_provider());
|
2018-06-22 15:09:15 +02:00
|
|
|
ap.insert_account(s0.secret().clone(), &"".into()).unwrap();
|
|
|
|
ap.insert_account(s1.secret().clone(), &"".into()).unwrap();
|
2017-01-18 18:49:50 +01:00
|
|
|
|
2017-08-21 13:46:58 +02:00
|
|
|
let chain_id = Spec::new_test_round().chain_id();
|
2018-04-20 15:45:53 +02:00
|
|
|
let mut net = TestNet::with_spec_and_accounts(2, SyncConfig::default(), Spec::new_test_round, Some(ap));
|
2018-04-09 16:14:33 +02:00
|
|
|
let io_handler0: Arc<IoHandler<ClientIoMessage>> = Arc::new(TestIoHandler::new(net.peer(0).chain.clone()));
|
|
|
|
let io_handler1: Arc<IoHandler<ClientIoMessage>> = Arc::new(TestIoHandler::new(net.peer(1).chain.clone()));
|
2016-12-11 12:32:01 +01:00
|
|
|
// Push transaction to both clients. Only one of them gets lucky to produce a block.
|
2018-04-13 17:34:27 +02:00
|
|
|
net.peer(0).miner.set_author(s0.address(), Some("".into())).unwrap();
|
2018-06-22 15:09:15 +02:00
|
|
|
net.peer(1).miner.set_author(s1.address(), Some("".into())).unwrap();
|
2017-09-05 17:54:05 +02:00
|
|
|
net.peer(0).chain.engine().register_client(Arc::downgrade(&net.peer(0).chain) as _);
|
|
|
|
net.peer(1).chain.engine().register_client(Arc::downgrade(&net.peer(1).chain) as _);
|
2016-12-11 12:32:01 +01:00
|
|
|
net.peer(0).chain.set_io_channel(IoChannel::to_handler(Arc::downgrade(&io_handler1)));
|
2016-12-12 17:24:48 +01:00
|
|
|
net.peer(1).chain.set_io_channel(IoChannel::to_handler(Arc::downgrade(&io_handler0)));
|
2016-12-11 12:32:01 +01:00
|
|
|
// exchange statuses
|
|
|
|
net.sync();
|
2016-12-12 17:24:48 +01:00
|
|
|
// Trigger block proposal
|
2018-04-13 17:34:27 +02:00
|
|
|
net.peer(0).miner.import_own_transaction(&*net.peer(0).chain, new_tx(s0.secret(), 0.into(), chain_id)).unwrap();
|
|
|
|
net.peer(1).miner.import_own_transaction(&*net.peer(1).chain, new_tx(s1.secret(), 0.into(), chain_id)).unwrap();
|
2016-12-12 17:24:48 +01:00
|
|
|
// Sync a block
|
2016-12-06 19:23:15 +01:00
|
|
|
net.sync();
|
|
|
|
assert_eq!(net.peer(0).chain.chain_info().best_block_number, 1);
|
|
|
|
assert_eq!(net.peer(1).chain.chain_info().best_block_number, 1);
|
|
|
|
|
2018-04-13 17:34:27 +02:00
|
|
|
net.peer(0).miner.import_own_transaction(&*net.peer(0).chain, new_tx(s0.secret(), 1.into(), chain_id)).unwrap();
|
|
|
|
net.peer(1).miner.import_own_transaction(&*net.peer(1).chain, new_tx(s1.secret(), 1.into(), chain_id)).unwrap();
|
2017-03-03 22:32:22 +01:00
|
|
|
// Move to next proposer step.
|
2016-12-12 17:24:48 +01:00
|
|
|
net.peer(0).chain.engine().step();
|
2016-12-06 19:23:15 +01:00
|
|
|
net.peer(1).chain.engine().step();
|
|
|
|
net.sync();
|
|
|
|
assert_eq!(net.peer(0).chain.chain_info().best_block_number, 2);
|
|
|
|
assert_eq!(net.peer(1).chain.chain_info().best_block_number, 2);
|
2016-12-12 17:24:48 +01:00
|
|
|
|
2017-03-03 22:32:22 +01:00
|
|
|
// Fork the network with equal height.
|
2018-04-13 17:34:27 +02:00
|
|
|
net.peer(0).miner.import_own_transaction(&*net.peer(0).chain, new_tx(s0.secret(), 2.into(), chain_id)).unwrap();
|
|
|
|
net.peer(1).miner.import_own_transaction(&*net.peer(1).chain, new_tx(s1.secret(), 2.into(), chain_id)).unwrap();
|
2017-03-03 22:32:22 +01:00
|
|
|
// Let both nodes build one block.
|
2016-12-12 17:24:48 +01:00
|
|
|
net.peer(0).chain.engine().step();
|
2017-03-03 22:32:22 +01:00
|
|
|
let early_hash = net.peer(0).chain.chain_info().best_block_hash;
|
2016-12-12 17:24:48 +01:00
|
|
|
net.peer(1).chain.engine().step();
|
|
|
|
net.peer(0).chain.engine().step();
|
|
|
|
net.peer(1).chain.engine().step();
|
2016-12-15 23:36:06 +01:00
|
|
|
let ci0 = net.peer(0).chain.chain_info();
|
|
|
|
let ci1 = net.peer(1).chain.chain_info();
|
|
|
|
assert_eq!(ci0.best_block_number, 3);
|
|
|
|
assert_eq!(ci1.best_block_number, 3);
|
|
|
|
assert!(ci0.best_block_hash != ci1.best_block_hash);
|
2017-03-03 22:32:22 +01:00
|
|
|
// Reorg to the chain with earlier view.
|
2016-12-12 17:24:48 +01:00
|
|
|
net.sync();
|
|
|
|
let ci0 = net.peer(0).chain.chain_info();
|
|
|
|
let ci1 = net.peer(1).chain.chain_info();
|
|
|
|
assert_eq!(ci0.best_block_number, 3);
|
|
|
|
assert_eq!(ci1.best_block_number, 3);
|
|
|
|
assert_eq!(ci0.best_block_hash, ci1.best_block_hash);
|
2017-03-03 22:32:22 +01:00
|
|
|
assert_eq!(ci1.best_block_hash, early_hash);
|
|
|
|
|
|
|
|
// Selfish miner
|
2018-04-13 17:34:27 +02:00
|
|
|
net.peer(0).miner.import_own_transaction(&*net.peer(0).chain, new_tx(s0.secret(), 3.into(), chain_id)).unwrap();
|
|
|
|
net.peer(1).miner.import_own_transaction(&*net.peer(1).chain, new_tx(s1.secret(), 3.into(), chain_id)).unwrap();
|
2017-03-03 22:32:22 +01:00
|
|
|
// Node 0 is an earlier primary.
|
|
|
|
net.peer(0).chain.engine().step();
|
|
|
|
assert_eq!(net.peer(0).chain.chain_info().best_block_number, 4);
|
|
|
|
net.peer(0).chain.engine().step();
|
|
|
|
net.peer(0).chain.engine().step();
|
|
|
|
net.peer(0).chain.engine().step();
|
|
|
|
assert_eq!(net.peer(0).chain.chain_info().best_block_number, 4);
|
|
|
|
// Node 1 makes 2 blocks, but is a later primary on the first one.
|
|
|
|
net.peer(1).chain.engine().step();
|
|
|
|
net.peer(1).chain.engine().step();
|
2018-04-13 17:34:27 +02:00
|
|
|
net.peer(1).miner.import_own_transaction(&*net.peer(1).chain, new_tx(s1.secret(), 4.into(), chain_id)).unwrap();
|
2017-03-03 22:32:22 +01:00
|
|
|
net.peer(1).chain.engine().step();
|
|
|
|
net.peer(1).chain.engine().step();
|
|
|
|
assert_eq!(net.peer(1).chain.chain_info().best_block_number, 5);
|
|
|
|
// Reorg to the longest chain one not ealier view one.
|
|
|
|
net.sync();
|
|
|
|
let ci0 = net.peer(0).chain.chain_info();
|
|
|
|
let ci1 = net.peer(1).chain.chain_info();
|
|
|
|
assert_eq!(ci0.best_block_number, 5);
|
|
|
|
assert_eq!(ci1.best_block_number, 5);
|
|
|
|
assert_eq!(ci0.best_block_hash, ci1.best_block_hash);
|
2016-12-06 19:23:15 +01:00
|
|
|
}
|
|
|
|
|
2016-12-10 17:40:53 +01:00
|
|
|
#[test]
|
2016-12-11 12:32:01 +01:00
|
|
|
fn tendermint() {
|
2017-08-31 11:35:41 +02:00
|
|
|
let s0 = KeyPair::from_secret_slice(&keccak("1")).unwrap();
|
|
|
|
let s1 = KeyPair::from_secret_slice(&keccak("0")).unwrap();
|
2017-01-18 18:49:50 +01:00
|
|
|
let ap = Arc::new(AccountProvider::transient_provider());
|
2018-06-22 15:09:15 +02:00
|
|
|
ap.insert_account(s0.secret().clone(), &"".into()).unwrap();
|
|
|
|
ap.insert_account(s1.secret().clone(), &"".into()).unwrap();
|
2017-01-18 18:49:50 +01:00
|
|
|
|
2017-08-21 13:46:58 +02:00
|
|
|
let chain_id = Spec::new_test_tendermint().chain_id();
|
2018-04-20 15:45:53 +02:00
|
|
|
let mut net = TestNet::with_spec_and_accounts(2, SyncConfig::default(), Spec::new_test_tendermint, Some(ap));
|
2018-04-09 16:14:33 +02:00
|
|
|
let io_handler0: Arc<IoHandler<ClientIoMessage>> = Arc::new(TestIoHandler::new(net.peer(0).chain.clone()));
|
|
|
|
let io_handler1: Arc<IoHandler<ClientIoMessage>> = Arc::new(TestIoHandler::new(net.peer(1).chain.clone()));
|
2016-12-10 17:40:53 +01:00
|
|
|
// Push transaction to both clients. Only one of them issues a proposal.
|
2018-04-13 17:34:27 +02:00
|
|
|
net.peer(0).miner.set_author(s0.address(), Some("".into())).unwrap();
|
2016-12-12 17:24:48 +01:00
|
|
|
trace!(target: "poa", "Peer 0 is {}.", s0.address());
|
2018-04-13 17:34:27 +02:00
|
|
|
net.peer(1).miner.set_author(s1.address(), Some("".into())).unwrap();
|
2016-12-12 17:24:48 +01:00
|
|
|
trace!(target: "poa", "Peer 1 is {}.", s1.address());
|
2017-09-05 17:54:05 +02:00
|
|
|
net.peer(0).chain.engine().register_client(Arc::downgrade(&net.peer(0).chain) as _);
|
|
|
|
net.peer(1).chain.engine().register_client(Arc::downgrade(&net.peer(1).chain) as _);
|
2016-12-12 17:24:48 +01:00
|
|
|
net.peer(0).chain.set_io_channel(IoChannel::to_handler(Arc::downgrade(&io_handler0)));
|
2016-12-11 12:32:01 +01:00
|
|
|
net.peer(1).chain.set_io_channel(IoChannel::to_handler(Arc::downgrade(&io_handler1)));
|
2016-12-12 17:24:48 +01:00
|
|
|
// Exhange statuses
|
2016-12-11 18:23:54 +01:00
|
|
|
net.sync();
|
2016-12-10 17:40:53 +01:00
|
|
|
// Propose
|
2018-04-13 17:34:27 +02:00
|
|
|
net.peer(0).miner.import_own_transaction(&*net.peer(0).chain, new_tx(s0.secret(), 0.into(), chain_id)).unwrap();
|
2016-12-10 17:40:53 +01:00
|
|
|
net.sync();
|
2016-12-12 17:24:48 +01:00
|
|
|
// Propose timeout, synchronous for now
|
2016-12-10 17:40:53 +01:00
|
|
|
net.peer(0).chain.engine().step();
|
2016-12-12 17:24:48 +01:00
|
|
|
net.peer(1).chain.engine().step();
|
|
|
|
// Prevote, precommit and commit
|
|
|
|
net.sync();
|
2016-12-10 17:40:53 +01:00
|
|
|
|
|
|
|
assert_eq!(net.peer(0).chain.chain_info().best_block_number, 1);
|
|
|
|
assert_eq!(net.peer(1).chain.chain_info().best_block_number, 1);
|
|
|
|
|
2018-04-13 17:34:27 +02:00
|
|
|
net.peer(1).miner.import_own_transaction(&*net.peer(1).chain, new_tx(s1.secret(), 0.into(), chain_id)).unwrap();
|
2016-12-12 17:24:48 +01:00
|
|
|
// Commit timeout
|
|
|
|
net.peer(0).chain.engine().step();
|
|
|
|
net.peer(1).chain.engine().step();
|
|
|
|
// Propose
|
|
|
|
net.sync();
|
|
|
|
// Propose timeout
|
|
|
|
net.peer(0).chain.engine().step();
|
2016-12-10 17:40:53 +01:00
|
|
|
net.peer(1).chain.engine().step();
|
2016-12-12 17:24:48 +01:00
|
|
|
// Prevote, precommit and commit
|
2016-12-10 17:40:53 +01:00
|
|
|
net.sync();
|
|
|
|
assert_eq!(net.peer(0).chain.chain_info().best_block_number, 2);
|
|
|
|
assert_eq!(net.peer(1).chain.chain_info().best_block_number, 2);
|
2016-12-12 17:24:48 +01:00
|
|
|
|
2018-04-13 17:34:27 +02:00
|
|
|
net.peer(0).miner.import_own_transaction(&*net.peer(0).chain, new_tx(s0.secret(), 1.into(), chain_id)).unwrap();
|
|
|
|
net.peer(1).miner.import_own_transaction(&*net.peer(1).chain, new_tx(s1.secret(), 1.into(), chain_id)).unwrap();
|
2016-12-12 17:24:48 +01:00
|
|
|
// Peers get disconnected.
|
|
|
|
// Commit
|
|
|
|
net.peer(0).chain.engine().step();
|
|
|
|
net.peer(1).chain.engine().step();
|
2016-12-12 19:44:24 +01:00
|
|
|
// Propose
|
2016-12-12 17:24:48 +01:00
|
|
|
net.peer(0).chain.engine().step();
|
|
|
|
net.peer(1).chain.engine().step();
|
2018-04-13 17:34:27 +02:00
|
|
|
net.peer(0).miner.import_own_transaction(&*net.peer(0).chain, new_tx(s0.secret(), 2.into(), chain_id)).unwrap();
|
|
|
|
net.peer(1).miner.import_own_transaction(&*net.peer(1).chain, new_tx(s1.secret(), 2.into(), chain_id)).unwrap();
|
2016-12-12 19:44:24 +01:00
|
|
|
// Send different prevotes
|
2016-12-12 17:24:48 +01:00
|
|
|
net.sync();
|
2016-12-12 19:44:24 +01:00
|
|
|
// Prevote timeout
|
2016-12-12 17:24:48 +01:00
|
|
|
net.peer(0).chain.engine().step();
|
|
|
|
net.peer(1).chain.engine().step();
|
2016-12-12 19:44:24 +01:00
|
|
|
// Precommit and commit
|
2016-12-12 17:24:48 +01:00
|
|
|
net.sync();
|
2016-12-12 19:44:24 +01:00
|
|
|
// Propose timeout
|
2016-12-12 17:24:48 +01:00
|
|
|
net.peer(0).chain.engine().step();
|
|
|
|
net.peer(1).chain.engine().step();
|
|
|
|
net.sync();
|
|
|
|
let ci0 = net.peer(0).chain.chain_info();
|
|
|
|
let ci1 = net.peer(1).chain.chain_info();
|
|
|
|
assert_eq!(ci0.best_block_number, 3);
|
|
|
|
assert_eq!(ci1.best_block_number, 3);
|
|
|
|
assert_eq!(ci0.best_block_hash, ci1.best_block_hash);
|
2016-12-10 17:40:53 +01:00
|
|
|
}
|