174 lines
6.1 KiB
Rust
174 lines
6.1 KiB
Rust
|
// Copyright 2015-2017 Parity Technologies (UK) Ltd.
|
||
|
// 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/>.
|
||
|
|
||
|
use ethkey::Signature;
|
||
|
use bytes::Bytes;
|
||
|
use std::collections::HashMap;
|
||
|
use ethereum_types::{H256, U256, Address};
|
||
|
use transaction::{UnverifiedTransaction, SignedTransaction};
|
||
|
use ethcore_miner::transaction_queue::{TransactionQueue, RemovalReason,
|
||
|
TransactionDetailsProvider as TransactionQueueDetailsProvider, TransactionOrigin};
|
||
|
use error::{Error, ErrorKind};
|
||
|
use ethcore::header::BlockNumber;
|
||
|
|
||
|
/// Maximum length for private transactions queues.
|
||
|
const MAX_QUEUE_LEN: usize = 8312;
|
||
|
|
||
|
/// Desriptor for private transaction stored in queue for verification
|
||
|
#[derive(Default, Debug, Clone, PartialEq, Eq)]
|
||
|
pub struct PrivateTransactionDesc {
|
||
|
/// Hash of the private transaction
|
||
|
pub private_hash: H256,
|
||
|
/// Contract's address used in private transaction
|
||
|
pub contract: Address,
|
||
|
/// Address that should be used for verification
|
||
|
pub validator_account: Address,
|
||
|
}
|
||
|
|
||
|
/// Storage for private transactions for verification
|
||
|
#[derive(Default)]
|
||
|
pub struct VerificationStore {
|
||
|
/// Descriptors for private transactions in queue for verification with key - hash of the original transaction
|
||
|
descriptors: HashMap<H256, PrivateTransactionDesc>,
|
||
|
/// Queue with transactions for verification
|
||
|
transactions: TransactionQueue,
|
||
|
}
|
||
|
|
||
|
impl VerificationStore {
|
||
|
/// Adds private transaction for verification into the store
|
||
|
pub fn add_transaction(
|
||
|
&mut self,
|
||
|
transaction: UnverifiedTransaction,
|
||
|
contract: Address,
|
||
|
validator_account: Address,
|
||
|
private_hash: H256,
|
||
|
details_provider: &TransactionQueueDetailsProvider,
|
||
|
insertion_time: BlockNumber,
|
||
|
) -> Result<(), Error> {
|
||
|
if self.descriptors.len() > MAX_QUEUE_LEN {
|
||
|
bail!(ErrorKind::QueueIsFull);
|
||
|
}
|
||
|
|
||
|
if self.descriptors.get(&transaction.hash()).is_some() {
|
||
|
bail!(ErrorKind::PrivateTransactionAlreadyImported);
|
||
|
}
|
||
|
let transaction_hash = transaction.hash();
|
||
|
let signed_transaction = SignedTransaction::new(transaction)?;
|
||
|
self.transactions
|
||
|
.add(signed_transaction, TransactionOrigin::External, insertion_time, None, details_provider)
|
||
|
.and_then(|_| {
|
||
|
self.descriptors.insert(transaction_hash, PrivateTransactionDesc{
|
||
|
private_hash,
|
||
|
contract,
|
||
|
validator_account,
|
||
|
});
|
||
|
Ok(())
|
||
|
})
|
||
|
.map_err(Into::into)
|
||
|
}
|
||
|
|
||
|
/// Returns transactions ready for verification
|
||
|
/// Returns only one transaction per sender because several cannot be verified in a row without verification from other peers
|
||
|
pub fn ready_transactions(&self) -> Vec<SignedTransaction> {
|
||
|
// TODO [ToDr] Performance killer, re-work with new transaction queue.
|
||
|
let mut transactions = self.transactions.top_transactions();
|
||
|
// TODO [ToDr] Potential issue (create low address to have your transactions processed first)
|
||
|
transactions.sort_by(|a, b| a.sender().cmp(&b.sender()));
|
||
|
transactions.dedup_by(|a, b| a.sender().eq(&b.sender()));
|
||
|
transactions
|
||
|
}
|
||
|
|
||
|
/// Returns descriptor of the corresponding private transaction
|
||
|
pub fn private_transaction_descriptor(&self, transaction_hash: &H256) -> Result<&PrivateTransactionDesc, Error> {
|
||
|
self.descriptors.get(transaction_hash).ok_or(ErrorKind::PrivateTransactionNotFound.into())
|
||
|
}
|
||
|
|
||
|
/// Remove transaction from the queue for verification
|
||
|
pub fn remove_private_transaction<F>(&mut self, transaction_hash: &H256, fetch_nonce: &F)
|
||
|
where F: Fn(&Address) -> U256 {
|
||
|
|
||
|
self.descriptors.remove(transaction_hash);
|
||
|
self.transactions.remove(transaction_hash, fetch_nonce, RemovalReason::Invalid);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/// Desriptor for private transaction stored in queue for signing
|
||
|
#[derive(Debug, Clone)]
|
||
|
pub struct PrivateTransactionSigningDesc {
|
||
|
/// Original unsigned transaction
|
||
|
pub original_transaction: SignedTransaction,
|
||
|
/// Supposed validators from the contract
|
||
|
pub validators: Vec<Address>,
|
||
|
/// Already obtained signatures
|
||
|
pub received_signatures: Vec<Signature>,
|
||
|
/// State after transaction execution to compare further with received from validators
|
||
|
pub state: Bytes,
|
||
|
/// Build-in nonce of the contract
|
||
|
pub contract_nonce: U256,
|
||
|
}
|
||
|
|
||
|
/// Storage for private transactions for signing
|
||
|
#[derive(Default)]
|
||
|
pub struct SigningStore {
|
||
|
/// Transactions and descriptors for signing
|
||
|
transactions: HashMap<H256, PrivateTransactionSigningDesc>,
|
||
|
}
|
||
|
|
||
|
impl SigningStore {
|
||
|
/// Adds new private transaction into the store for signing
|
||
|
pub fn add_transaction(
|
||
|
&mut self,
|
||
|
private_hash: H256,
|
||
|
transaction: SignedTransaction,
|
||
|
validators: Vec<Address>,
|
||
|
state: Bytes,
|
||
|
contract_nonce: U256,
|
||
|
) -> Result<(), Error> {
|
||
|
if self.transactions.len() > MAX_QUEUE_LEN {
|
||
|
bail!(ErrorKind::QueueIsFull);
|
||
|
}
|
||
|
|
||
|
self.transactions.insert(private_hash, PrivateTransactionSigningDesc {
|
||
|
original_transaction: transaction.clone(),
|
||
|
validators: validators.clone(),
|
||
|
received_signatures: Vec::new(),
|
||
|
state,
|
||
|
contract_nonce,
|
||
|
});
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
/// Get copy of private transaction's description from the storage
|
||
|
pub fn get(&self, private_hash: &H256) -> Option<PrivateTransactionSigningDesc> {
|
||
|
self.transactions.get(private_hash).cloned()
|
||
|
}
|
||
|
|
||
|
/// Removes desc from the store (after verification is completed)
|
||
|
pub fn remove(&mut self, private_hash: &H256) -> Result<(), Error> {
|
||
|
self.transactions.remove(private_hash);
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
/// Adds received signature for the stored private transaction
|
||
|
pub fn add_signature(&mut self, private_hash: &H256, signature: Signature) -> Result<(), Error> {
|
||
|
let desc = self.transactions.get_mut(private_hash).ok_or_else(|| ErrorKind::PrivateTransactionNotFound)?;
|
||
|
if !desc.received_signatures.contains(&signature) {
|
||
|
desc.received_signatures.push(signature);
|
||
|
}
|
||
|
Ok(())
|
||
|
}
|
||
|
}
|