propose message test
This commit is contained in:
		
							parent
							
								
									77f06be7fb
								
							
						
					
					
						commit
						fcae03e55f
					
				
							
								
								
									
										42
									
								
								ethcore/res/tendermint.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								ethcore/res/tendermint.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,42 @@ | |||||||
|  | { | ||||||
|  | 	"name": "TestBFT", | ||||||
|  | 	"engine": { | ||||||
|  | 		"Tendermint": { | ||||||
|  | 			"params": { | ||||||
|  | 				"gasLimitBoundDivisor": "0x0400", | ||||||
|  | 				"durationLimit": "0x0d", | ||||||
|  | 				"validators" : [ | ||||||
|  | 					"0x7d577a597b2742b498cb5cf0c26cdcd726d39e6e", | ||||||
|  | 					"0x82a978b3f5962a5b0957d9ee9eef472ee55b42f1" | ||||||
|  | 				] | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	}, | ||||||
|  | 	"params": { | ||||||
|  | 		"accountStartNonce": "0x0100000", | ||||||
|  | 		"maximumExtraDataSize": "0x20", | ||||||
|  | 		"minGasLimit": "0x1388", | ||||||
|  | 		"networkID" : "0x69" | ||||||
|  | 	}, | ||||||
|  | 	"genesis": { | ||||||
|  | 		"seal": { | ||||||
|  | 			"generic": { | ||||||
|  | 				"fields": 1, | ||||||
|  | 				"rlp": "0x11bbe8db4e347b4e8c937c1c8370e4b5ed33adb3db69cbdb7a38e1e50b1b82fa" | ||||||
|  | 			} | ||||||
|  | 		}, | ||||||
|  | 		"difficulty": "0x20000", | ||||||
|  | 		"author": "0x0000000000000000000000000000000000000000", | ||||||
|  | 		"timestamp": "0x00", | ||||||
|  | 		"parentHash": "0x0000000000000000000000000000000000000000000000000000000000000000", | ||||||
|  | 		"extraData": "0x", | ||||||
|  | 		"gasLimit": "0x2fefd8" | ||||||
|  | 	}, | ||||||
|  | 	"accounts": { | ||||||
|  | 		"0000000000000000000000000000000000000001": { "balance": "1", "nonce": "1048576", "builtin": { "name": "ecrecover", "pricing": { "linear": { "base": 3000, "word": 0 } } } }, | ||||||
|  | 		"0000000000000000000000000000000000000002": { "balance": "1", "nonce": "1048576", "builtin": { "name": "sha256", "pricing": { "linear": { "base": 60, "word": 12 } } } }, | ||||||
|  | 		"0000000000000000000000000000000000000003": { "balance": "1", "nonce": "1048576", "builtin": { "name": "ripemd160", "pricing": { "linear": { "base": 600, "word": 120 } } } }, | ||||||
|  | 		"0000000000000000000000000000000000000004": { "balance": "1", "nonce": "1048576", "builtin": { "name": "identity", "pricing": { "linear": { "base": 15, "word": 3 } } } }, | ||||||
|  | 		"9cce34f7ab185c7aba1b7c8140d620b4bda941d6": { "balance": "1606938044258990275541962092341162602522202993782792835301376", "nonce": "1048576" } | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @ -1022,6 +1022,7 @@ impl BlockChainClient for Client { | |||||||
| 		self.miner.pending_transactions() | 		self.miner.pending_transactions() | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	// TODO: Make it an actual queue, return errors.
 | ||||||
| 	fn queue_infinity_message(&self, message: Bytes) { | 	fn queue_infinity_message(&self, message: Bytes) { | ||||||
| 		let full_rlp = UntrustedRlp::new(&message); | 		let full_rlp = UntrustedRlp::new(&message); | ||||||
| 		if let Ok(signature) = full_rlp.val_at(0) { | 		if let Ok(signature) = full_rlp.val_at(0) { | ||||||
|  | |||||||
| @ -38,7 +38,7 @@ use spec::Spec; | |||||||
| use block_queue::BlockQueueInfo; | use block_queue::BlockQueueInfo; | ||||||
| use block::{OpenBlock, SealedBlock}; | use block::{OpenBlock, SealedBlock}; | ||||||
| use executive::Executed; | use executive::Executed; | ||||||
| use error::{Error, CallError}; | use error::CallError; | ||||||
| use trace::LocalizedTrace; | use trace::LocalizedTrace; | ||||||
| 
 | 
 | ||||||
| /// Test client.
 | /// Test client.
 | ||||||
|  | |||||||
| @ -134,7 +134,7 @@ pub trait Engine : Sync + Send { | |||||||
| 
 | 
 | ||||||
| 	/// Handle any potential consensus messages;
 | 	/// Handle any potential consensus messages;
 | ||||||
| 	/// updating consensus state and potentially issuing a new one.
 | 	/// updating consensus state and potentially issuing a new one.
 | ||||||
| 	fn handle_message(&self, sender: Address, message: UntrustedRlp) -> Result<Bytes, Error> { Err(EngineError::UnexpectedMessage.into()) } | 	fn handle_message(&self, _sender: Address, _message: UntrustedRlp) -> Result<Bytes, Error> { Err(EngineError::UnexpectedMessage.into()) } | ||||||
| 
 | 
 | ||||||
| 	// TODO: builtin contract routing - to do this properly, it will require removing the built-in configuration-reading logic
 | 	// TODO: builtin contract routing - to do this properly, it will require removing the built-in configuration-reading logic
 | ||||||
| 	// from Spec into here and removing the Spec::builtins field.
 | 	// from Spec into here and removing the Spec::builtins field.
 | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ | |||||||
| 
 | 
 | ||||||
| //! Tendermint BFT consensus engine with round robin proof-of-authority.
 | //! Tendermint BFT consensus engine with round robin proof-of-authority.
 | ||||||
| 
 | 
 | ||||||
|  | use std::sync::atomic::{AtomicUsize, Ordering as AtomicOrdering}; | ||||||
| use common::*; | use common::*; | ||||||
| use account_provider::AccountProvider; | use account_provider::AccountProvider; | ||||||
| use block::*; | use block::*; | ||||||
| @ -40,7 +41,7 @@ pub struct TendermintParams { | |||||||
| 	/// Consensus step.
 | 	/// Consensus step.
 | ||||||
| 	s: RwLock<Step>, | 	s: RwLock<Step>, | ||||||
| 	/// Used to swith proposer.
 | 	/// Used to swith proposer.
 | ||||||
| 	proposer_nonce: usize | 	proposer_nonce: AtomicUsize | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| @ -62,7 +63,7 @@ impl From<ethjson::spec::TendermintParams> for TendermintParams { | |||||||
| 			validator_n: val_n, | 			validator_n: val_n, | ||||||
| 			r: 0, | 			r: 0, | ||||||
| 			s: RwLock::new(Step::Propose), | 			s: RwLock::new(Step::Propose), | ||||||
| 			proposer_nonce: 0 | 			proposer_nonce: AtomicUsize::new(0) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -86,7 +87,7 @@ impl Tendermint { | |||||||
| 
 | 
 | ||||||
| 	fn proposer(&self) -> Address { | 	fn proposer(&self) -> Address { | ||||||
| 		let ref p = self.our_params; | 		let ref p = self.our_params; | ||||||
| 		p.validators.get(p.proposer_nonce%p.validator_n).unwrap().clone() | 		p.validators.get(p.proposer_nonce.load(AtomicOrdering::Relaxed)%p.validator_n).unwrap().clone() | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	fn propose_message(&self, message: UntrustedRlp) -> Result<Bytes, Error> { | 	fn propose_message(&self, message: UntrustedRlp) -> Result<Bytes, Error> { | ||||||
| @ -94,7 +95,8 @@ impl Tendermint { | |||||||
| 			Step::Propose => (), | 			Step::Propose => (), | ||||||
| 			_ => try!(Err(EngineError::WrongStep)), | 			_ => try!(Err(EngineError::WrongStep)), | ||||||
| 		} | 		} | ||||||
| 		let proposal = try!(message.val_at(0)); | 		let proposal = try!(message.as_val()); | ||||||
|  | 		self.our_params.proposer_nonce.fetch_add(1, AtomicOrdering::Relaxed); | ||||||
| 		let vote = ProposeCollect::new(proposal, | 		let vote = ProposeCollect::new(proposal, | ||||||
| 									   self.our_params.validators.iter().cloned().collect(), | 									   self.our_params.validators.iter().cloned().collect(), | ||||||
| 									   self.threshold()); | 									   self.threshold()); | ||||||
| @ -164,8 +166,8 @@ impl Engine for Tendermint { | |||||||
| 
 | 
 | ||||||
| 	fn handle_message(&self, sender: Address, message: UntrustedRlp) -> Result<Bytes, Error> { | 	fn handle_message(&self, sender: Address, message: UntrustedRlp) -> Result<Bytes, Error> { | ||||||
| 		match try!(message.val_at(0)) { | 		match try!(message.val_at(0)) { | ||||||
| 			0u8 if sender == self.proposer() => self.propose_message(message), | 			0u8 if sender == self.proposer() => self.propose_message(try!(message.at(1))), | ||||||
| 			1 => self.prevote_message(sender, message), | 			1 => self.prevote_message(sender, try!(message.at(1))), | ||||||
| 			_ => try!(Err(EngineError::UnknownStep)), | 			_ => try!(Err(EngineError::UnknownStep)), | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -222,8 +224,10 @@ mod tests { | |||||||
| 	use tests::helpers::*; | 	use tests::helpers::*; | ||||||
| 	use account_provider::AccountProvider; | 	use account_provider::AccountProvider; | ||||||
| 	use spec::Spec; | 	use spec::Spec; | ||||||
|  | 	use super::Step; | ||||||
| 
 | 
 | ||||||
| 	/// Create a new test chain spec with `Tendermint` consensus engine.
 | 	/// Create a new test chain spec with `Tendermint` consensus engine.
 | ||||||
|  | 	/// Account "0".sha3() and "1".sha3() are a validators.
 | ||||||
| 	fn new_test_tendermint() -> Spec { Spec::load(include_bytes!("../../res/tendermint.json")) } | 	fn new_test_tendermint() -> Spec { Spec::load(include_bytes!("../../res/tendermint.json")) } | ||||||
| 
 | 
 | ||||||
| 	#[test] | 	#[test] | ||||||
| @ -302,9 +306,23 @@ mod tests { | |||||||
| 	fn propose_step() { | 	fn propose_step() { | ||||||
| 		let engine = new_test_tendermint().engine; | 		let engine = new_test_tendermint().engine; | ||||||
| 		let tap = AccountProvider::transient_provider(); | 		let tap = AccountProvider::transient_provider(); | ||||||
| 		let addr = tap.insert_account("1".sha3(), "1").unwrap(); | 		let mut s = RlpStream::new_list(2); | ||||||
| 		println!("{:?}", addr); | 		let header = Header::default(); | ||||||
| 		false; | 		s.append(&0u8).append(&header.bare_hash()); | ||||||
|  | 		let drain = s.out(); | ||||||
|  | 		let propose_rlp = UntrustedRlp::new(&drain); | ||||||
|  | 
 | ||||||
|  | 		let not_validator_addr = tap.insert_account("101".sha3(), "101").unwrap(); | ||||||
|  | 		assert!(engine.handle_message(not_validator_addr, propose_rlp.clone()).is_err()); | ||||||
|  | 
 | ||||||
|  | 		let not_proposer_addr = tap.insert_account("0".sha3(), "0").unwrap(); | ||||||
|  | 		assert!(engine.handle_message(not_proposer_addr, propose_rlp.clone()).is_err()); | ||||||
|  | 
 | ||||||
|  | 		let proposer_addr = tap.insert_account("1".sha3(), "1").unwrap(); | ||||||
|  | 		assert_eq!(vec![160, 39, 191, 179, 126, 80, 124, 233, 13, 161, 65, 48, 114, 4, 177, 198, 186, 36, 25, 67, 128, 97, 53, 144, 172, 80, 202, 75, 29, 113, 152, 255, 101], | ||||||
|  | 				   engine.handle_message(proposer_addr, propose_rlp.clone()).unwrap()); | ||||||
|  | 
 | ||||||
|  | 		assert!(engine.handle_message(not_proposer_addr, propose_rlp).is_err()); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	#[test] | 	#[test] | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user