2019-01-07 11:33:07 +01:00
|
|
|
// Copyright 2015-2019 Parity Technologies (UK) Ltd.
|
|
|
|
// This file is part of Parity Ethereum.
|
2017-12-20 14:02:21 +01:00
|
|
|
|
2019-01-07 11:33:07 +01:00
|
|
|
// Parity Ethereum is free software: you can redistribute it and/or modify
|
2017-12-20 14:02:21 +01:00
|
|
|
// 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.
|
|
|
|
|
2019-01-07 11:33:07 +01:00
|
|
|
// Parity Ethereum is distributed in the hope that it will be useful,
|
2017-12-20 14:02:21 +01:00
|
|
|
// 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
|
2019-01-07 11:33:07 +01:00
|
|
|
// along with Parity Ethereum. If not, see <http://www.gnu.org/licenses/>.
|
2017-12-20 14:02:21 +01:00
|
|
|
|
|
|
|
use std::sync::{Arc, Weak};
|
2018-04-09 16:38:59 +02:00
|
|
|
use bytes::Bytes;
|
2019-01-17 16:43:08 +01:00
|
|
|
use call_contract::RegistryInfo;
|
2019-08-13 12:33:34 +02:00
|
|
|
use common_types::{
|
|
|
|
ids::BlockId,
|
|
|
|
transaction::{Transaction, SignedTransaction, Action},
|
|
|
|
};
|
2018-04-09 16:38:59 +02:00
|
|
|
use ethereum_types::Address;
|
2019-08-15 17:59:22 +02:00
|
|
|
use ethcore::client::Client;
|
|
|
|
use client_traits::{ChainInfo, Nonce};
|
2018-04-09 16:38:59 +02:00
|
|
|
use ethcore::miner::{Miner, MinerService};
|
2018-04-10 12:13:49 +02:00
|
|
|
use sync::SyncProvider;
|
2018-06-14 09:01:52 +02:00
|
|
|
use helpers::{get_confirmed_block_hash, REQUEST_CONFIRMATIONS_REQUIRED};
|
|
|
|
use {Error, NodeKeyPair, ContractAddress};
|
2017-12-20 14:02:21 +01:00
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
/// 'Trusted' client weak reference.
|
|
|
|
pub struct TrustedClient {
|
2018-04-09 16:38:59 +02:00
|
|
|
/// This key server node key pair.
|
|
|
|
self_key_pair: Arc<NodeKeyPair>,
|
2017-12-20 14:02:21 +01:00
|
|
|
/// Blockchain client.
|
|
|
|
client: Weak<Client>,
|
|
|
|
/// Sync provider.
|
|
|
|
sync: Weak<SyncProvider>,
|
2018-04-09 16:38:59 +02:00
|
|
|
/// Miner service.
|
|
|
|
miner: Weak<Miner>,
|
2017-12-20 14:02:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl TrustedClient {
|
|
|
|
/// Create new trusted client.
|
2019-08-15 17:59:22 +02:00
|
|
|
pub fn new(self_key_pair: Arc<dyn NodeKeyPair>, client: Arc<Client>, sync: Arc<dyn SyncProvider>, miner: Arc<Miner>) -> Self {
|
2017-12-20 14:02:21 +01:00
|
|
|
TrustedClient {
|
2019-08-15 17:59:22 +02:00
|
|
|
self_key_pair,
|
2017-12-20 14:02:21 +01:00
|
|
|
client: Arc::downgrade(&client),
|
|
|
|
sync: Arc::downgrade(&sync),
|
2018-04-09 16:38:59 +02:00
|
|
|
miner: Arc::downgrade(&miner),
|
2017-12-20 14:02:21 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get 'trusted' `Client` reference only if it is synchronized && trusted.
|
|
|
|
pub fn get(&self) -> Option<Arc<Client>> {
|
|
|
|
self.client.upgrade()
|
|
|
|
.and_then(|client| self.sync.upgrade().map(|sync| (client, sync)))
|
|
|
|
.and_then(|(client, sync)| {
|
2019-04-02 17:13:55 +02:00
|
|
|
let is_synced = !sync.is_major_syncing();
|
2017-12-20 14:02:21 +01:00
|
|
|
let is_trusted = client.chain_info().security_level().is_full();
|
|
|
|
match is_synced && is_trusted {
|
|
|
|
true => Some(client),
|
|
|
|
false => None,
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get untrusted `Client` reference.
|
|
|
|
pub fn get_untrusted(&self) -> Option<Arc<Client>> {
|
|
|
|
self.client.upgrade()
|
|
|
|
}
|
2018-04-09 16:38:59 +02:00
|
|
|
|
|
|
|
/// Transact contract.
|
|
|
|
pub fn transact_contract(&self, contract: Address, tx_data: Bytes) -> Result<(), Error> {
|
|
|
|
let client = self.client.upgrade().ok_or_else(|| Error::Internal("cannot submit tx when client is offline".into()))?;
|
|
|
|
let miner = self.miner.upgrade().ok_or_else(|| Error::Internal("cannot submit tx when miner is offline".into()))?;
|
|
|
|
let engine = client.engine();
|
|
|
|
let transaction = Transaction {
|
|
|
|
nonce: client.latest_nonce(&self.self_key_pair.address()),
|
|
|
|
action: Action::Call(contract),
|
2018-04-13 17:34:27 +02:00
|
|
|
gas: miner.authoring_params().gas_range_target.0,
|
2018-04-09 16:38:59 +02:00
|
|
|
gas_price: miner.sensible_gas_price(),
|
|
|
|
value: Default::default(),
|
|
|
|
data: tx_data,
|
|
|
|
};
|
|
|
|
let chain_id = engine.signing_chain_id(&client.latest_env_info());
|
|
|
|
let signature = self.self_key_pair.sign(&transaction.hash(chain_id))?;
|
|
|
|
let signed = SignedTransaction::new(transaction.with_signature(signature, chain_id))?;
|
|
|
|
miner.import_own_transaction(&*client, signed.into())
|
|
|
|
.map_err(|e| Error::Internal(format!("failed to import tx: {}", e)))
|
2018-06-14 09:01:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Read contract address. If address source is registry, address only returned if current client state is
|
|
|
|
/// trusted. Address from registry is read from registry from block latest block with
|
|
|
|
/// REQUEST_CONFIRMATIONS_REQUIRED confirmations.
|
|
|
|
pub fn read_contract_address(&self, registry_name: String, address: &ContractAddress) -> Option<Address> {
|
|
|
|
match *address {
|
|
|
|
ContractAddress::Address(ref address) => Some(address.clone()),
|
|
|
|
ContractAddress::Registry => self.get().and_then(|client|
|
|
|
|
get_confirmed_block_hash(&*client, REQUEST_CONFIRMATIONS_REQUIRED)
|
|
|
|
.and_then(|block| client.registry_address(registry_name, BlockId::Hash(block)))
|
|
|
|
),
|
|
|
|
}
|
2018-04-09 16:38:59 +02:00
|
|
|
}
|
2017-12-20 14:02:21 +01:00
|
|
|
}
|