Base for Signer Websockets server (#1158)
* Basic signing queue * Adding docs * WebSockets server for signer * Removing TODO * Shortening the syntax * Exposing types from RPC * Fixing indentation * Update main.rs
This commit is contained in:
		
							parent
							
								
									468d761e5c
								
							
						
					
					
						commit
						87d0f09a44
					
				
							
								
								
									
										33
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										33
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							| @ -105,6 +105,11 @@ name = "bloomchain" | |||||||
| version = "0.1.0" | version = "0.1.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "byteorder" | ||||||
|  | version = "0.5.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "bytes" | name = "bytes" | ||||||
| version = "0.3.0" | version = "0.3.0" | ||||||
| @ -347,9 +352,16 @@ version = "1.2.0" | |||||||
| dependencies = [ | dependencies = [ | ||||||
|  "clippy 0.0.69 (registry+https://github.com/rust-lang/crates.io-index)", |  "clippy 0.0.69 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  "env_logger 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", |  "env_logger 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "ethcore-rpc 1.2.0", | ||||||
|  "ethcore-util 1.2.0", |  "ethcore-util 1.2.0", | ||||||
|  "log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)", |  "log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "rustc-serialize 0.3.19 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  "rustc_version 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)", |  "rustc_version 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "serde 0.7.0 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "serde_codegen 0.7.4 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "serde_json 0.7.0 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "syntex 0.32.0 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "ws 0.4.7 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
| [[package]] | [[package]] | ||||||
| @ -1161,6 +1173,14 @@ dependencies = [ | |||||||
|  "serde 0.7.0 (registry+https://github.com/rust-lang/crates.io-index)", |  "serde 0.7.0 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "sha1" | ||||||
|  | version = "0.1.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | dependencies = [ | ||||||
|  |  "byteorder 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "sha3" | name = "sha3" | ||||||
| version = "0.1.0" | version = "0.1.0" | ||||||
| @ -1405,6 +1425,19 @@ name = "winapi-build" | |||||||
| version = "0.1.1" | version = "0.1.1" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "ws" | ||||||
|  | version = "0.4.7" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | dependencies = [ | ||||||
|  |  "httparse 1.1.2 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "mio 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "rand 0.3.14 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "sha1 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  |  "url 0.5.9 (registry+https://github.com/rust-lang/crates.io-index)", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "ws2_32-sys" | name = "ws2_32-sys" | ||||||
| version = "0.2.1" | version = "0.2.1" | ||||||
|  | |||||||
| @ -97,6 +97,11 @@ API and Console Options: | |||||||
|   --dapps-pass PASSWORD    Specify password for Dapps server. Use only in |   --dapps-pass PASSWORD    Specify password for Dapps server. Use only in | ||||||
|                            conjunction with --dapps-user. |                            conjunction with --dapps-user. | ||||||
| 
 | 
 | ||||||
|  |   --signer                 Enable Trusted Signer WebSocket endpoint used by | ||||||
|  |                            System UIs. | ||||||
|  |   --signer-port PORT       Specify the port of Trusted Signer server | ||||||
|  |                            [default: 8180]. | ||||||
|  | 
 | ||||||
| Sealing/Mining Options: | Sealing/Mining Options: | ||||||
|   --force-sealing          Force the node to author new blocks as if it were |   --force-sealing          Force the node to author new blocks as if it were | ||||||
|                            always sealing/mining. |                            always sealing/mining. | ||||||
| @ -234,6 +239,8 @@ pub struct Args { | |||||||
| 	pub flag_dapps_interface: String, | 	pub flag_dapps_interface: String, | ||||||
| 	pub flag_dapps_user: Option<String>, | 	pub flag_dapps_user: Option<String>, | ||||||
| 	pub flag_dapps_pass: Option<String>, | 	pub flag_dapps_pass: Option<String>, | ||||||
|  | 	pub flag_signer: bool, | ||||||
|  | 	pub flag_signer_port: u16, | ||||||
| 	pub flag_force_sealing: bool, | 	pub flag_force_sealing: bool, | ||||||
| 	pub flag_author: String, | 	pub flag_author: String, | ||||||
| 	pub flag_usd_per_tx: String, | 	pub flag_usd_per_tx: String, | ||||||
|  | |||||||
| @ -50,6 +50,9 @@ extern crate ethcore_rpc; | |||||||
| #[cfg(feature = "dapps")] | #[cfg(feature = "dapps")] | ||||||
| extern crate ethcore_dapps; | extern crate ethcore_dapps; | ||||||
| 
 | 
 | ||||||
|  | #[cfg(feature = "ethcore-signer")] | ||||||
|  | extern crate ethcore_signer; | ||||||
|  | 
 | ||||||
| #[macro_use] | #[macro_use] | ||||||
| mod die; | mod die; | ||||||
| mod price_info; | mod price_info; | ||||||
| @ -63,6 +66,7 @@ mod io_handler; | |||||||
| mod cli; | mod cli; | ||||||
| mod configuration; | mod configuration; | ||||||
| mod migration; | mod migration; | ||||||
|  | mod signer; | ||||||
| 
 | 
 | ||||||
| use std::io::{Write, Read, BufReader, BufRead}; | use std::io::{Write, Read, BufReader, BufRead}; | ||||||
| use std::ops::Deref; | use std::ops::Deref; | ||||||
| @ -89,6 +93,7 @@ use informant::Informant; | |||||||
| use die::*; | use die::*; | ||||||
| use cli::print_version; | use cli::print_version; | ||||||
| use rpc::RpcServer; | use rpc::RpcServer; | ||||||
|  | use signer::SignerServer; | ||||||
| use dapps::WebappServer; | use dapps::WebappServer; | ||||||
| use io_handler::ClientIoHandler; | use io_handler::ClientIoHandler; | ||||||
| use configuration::Configuration; | use configuration::Configuration; | ||||||
| @ -231,6 +236,14 @@ fn execute_client(conf: Configuration, spec: Spec, client_config: ClientConfig) | |||||||
| 		settings: network_settings.clone(), | 		settings: network_settings.clone(), | ||||||
| 	}); | 	}); | ||||||
| 
 | 
 | ||||||
|  | 	// Set up a signer
 | ||||||
|  | 	let signer_server = signer::start(signer::Configuration { | ||||||
|  | 		enabled: conf.args.flag_signer, | ||||||
|  | 		port: conf.args.flag_signer_port, | ||||||
|  | 	}, signer::Dependencies { | ||||||
|  | 		panic_handler: panic_handler.clone(), | ||||||
|  | 	}); | ||||||
|  | 
 | ||||||
| 	// Register IO handler
 | 	// Register IO handler
 | ||||||
| 	let io_handler  = Arc::new(ClientIoHandler { | 	let io_handler  = Arc::new(ClientIoHandler { | ||||||
| 		client: service.client(), | 		client: service.client(), | ||||||
| @ -241,7 +254,7 @@ fn execute_client(conf: Configuration, spec: Spec, client_config: ClientConfig) | |||||||
| 	service.io().register_handler(io_handler).expect("Error registering IO handler"); | 	service.io().register_handler(io_handler).expect("Error registering IO handler"); | ||||||
| 
 | 
 | ||||||
| 	// Handle exit
 | 	// Handle exit
 | ||||||
| 	wait_for_exit(panic_handler, rpc_server, dapps_server); | 	wait_for_exit(panic_handler, rpc_server, dapps_server, signer_server); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn flush_stdout() { | fn flush_stdout() { | ||||||
| @ -453,7 +466,12 @@ fn execute_account_cli(conf: Configuration) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn wait_for_exit(panic_handler: Arc<PanicHandler>, _rpc_server: Option<RpcServer>, _dapps_server: Option<WebappServer>) { | fn wait_for_exit( | ||||||
|  | 	panic_handler: Arc<PanicHandler>, | ||||||
|  | 	_rpc_server: Option<RpcServer>, | ||||||
|  | 	_dapps_server: Option<WebappServer>, | ||||||
|  | 	_signer_server: Option<SignerServer> | ||||||
|  | 	) { | ||||||
| 	let exit = Arc::new(Condvar::new()); | 	let exit = Arc::new(Condvar::new()); | ||||||
| 
 | 
 | ||||||
| 	// Handle possible exits
 | 	// Handle possible exits
 | ||||||
|  | |||||||
| @ -27,6 +27,8 @@ pub fn setup_log(init: &Option<String>) -> Arc<RotatingLogger> { | |||||||
| 
 | 
 | ||||||
| 	let mut levels = String::new(); | 	let mut levels = String::new(); | ||||||
| 	let mut builder = LogBuilder::new(); | 	let mut builder = LogBuilder::new(); | ||||||
|  | 	// Disable ws info logging by default.
 | ||||||
|  | 	builder.filter(Some("ws"), LogLevelFilter::Warn); | ||||||
| 	builder.filter(None, LogLevelFilter::Info); | 	builder.filter(None, LogLevelFilter::Info); | ||||||
| 
 | 
 | ||||||
| 	if env::var("RUST_LOG").is_ok() { | 	if env::var("RUST_LOG").is_ok() { | ||||||
|  | |||||||
							
								
								
									
										67
									
								
								parity/signer.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								parity/signer.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,67 @@ | |||||||
|  | // Copyright 2015, 2016 Ethcore (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 std::sync::Arc; | ||||||
|  | use util::panics::{PanicHandler, ForwardPanic}; | ||||||
|  | use die::*; | ||||||
|  | 
 | ||||||
|  | #[cfg(feature = "ethcore-signer")] | ||||||
|  | use ethcore_signer as signer; | ||||||
|  | #[cfg(feature = "ethcore-signer")] | ||||||
|  | pub use ethcore_signer::Server as SignerServer; | ||||||
|  | #[cfg(not(feature = "ethcore-signer"))] | ||||||
|  | pub struct SignerServer; | ||||||
|  | 
 | ||||||
|  | pub struct Configuration { | ||||||
|  | 	pub enabled: bool, | ||||||
|  | 	pub port: u16, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub struct Dependencies { | ||||||
|  | 	pub panic_handler: Arc<PanicHandler>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(feature = "ethcore-signer")] | ||||||
|  | pub fn start(conf: Configuration, deps: Dependencies) -> Option<SignerServer> { | ||||||
|  | 	if !conf.enabled { | ||||||
|  | 		return None; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	let addr = format!("127.0.0.1:{}", conf.port).parse().unwrap_or_else(|_| die!("Invalid port specified: {}", conf.port)); | ||||||
|  | 	let start_result = signer::Server::start(addr); | ||||||
|  | 
 | ||||||
|  | 	match start_result { | ||||||
|  | 		Err(signer::ServerError::IoError(err)) => die_with_io_error("Trusted Signer", err), | ||||||
|  | 		Err(e) => die!("Trusted Signer: {:?}", e), | ||||||
|  | 		Ok(server) => { | ||||||
|  | 			deps.panic_handler.forward_from(&server); | ||||||
|  | 			Some(server) | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(not(feature = "ethcore-signer"))] | ||||||
|  | pub fn start(conf: Configuration) -> Option<SignerServer> { | ||||||
|  | 	if !conf.enabled { | ||||||
|  | 		return None; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	die!("Your Parity version has been compiled without Trusted Signer support.") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| @ -18,12 +18,12 @@ | |||||||
| //!
 | //!
 | ||||||
| //! Compliant with ethereum rpc.
 | //! Compliant with ethereum rpc.
 | ||||||
| 
 | 
 | ||||||
| pub mod traits; |  | ||||||
| mod impls; | mod impls; | ||||||
| mod types; |  | ||||||
| mod helpers; | mod helpers; | ||||||
| 
 | 
 | ||||||
|  | pub mod traits; | ||||||
| pub mod tests; | pub mod tests; | ||||||
|  | pub mod types; | ||||||
| 
 | 
 | ||||||
| pub use self::traits::{Web3, Eth, EthFilter, Personal, Net, Ethcore, Traces, Rpc}; | pub use self::traits::{Web3, Eth, EthFilter, Personal, Net, Ethcore, Traces, Rpc}; | ||||||
| pub use self::impls::*; | pub use self::impls::*; | ||||||
|  | |||||||
| @ -18,9 +18,12 @@ use serde::{Serialize, Serializer}; | |||||||
| use util::numbers::*; | use util::numbers::*; | ||||||
| use v1::types::{Bytes, Transaction, OptionalValue}; | use v1::types::{Bytes, Transaction, OptionalValue}; | ||||||
| 
 | 
 | ||||||
|  | /// Block Transactions
 | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub enum BlockTransactions { | pub enum BlockTransactions { | ||||||
|  | 	/// Only hashes
 | ||||||
| 	Hashes(Vec<H256>), | 	Hashes(Vec<H256>), | ||||||
|  | 	/// Full transactions
 | ||||||
| 	Full(Vec<Transaction>) | 	Full(Vec<Transaction>) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -34,38 +37,58 @@ impl Serialize for BlockTransactions { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Block representation
 | ||||||
| #[derive(Debug, Serialize)] | #[derive(Debug, Serialize)] | ||||||
| pub struct Block { | pub struct Block { | ||||||
|  | 	/// Hash of the block
 | ||||||
| 	pub hash: OptionalValue<H256>, | 	pub hash: OptionalValue<H256>, | ||||||
|  | 	/// Hash of the parent
 | ||||||
| 	#[serde(rename="parentHash")] | 	#[serde(rename="parentHash")] | ||||||
| 	pub parent_hash: H256, | 	pub parent_hash: H256, | ||||||
|  | 	/// Hash of the uncles
 | ||||||
| 	#[serde(rename="sha3Uncles")] | 	#[serde(rename="sha3Uncles")] | ||||||
| 	pub uncles_hash: H256, | 	pub uncles_hash: H256, | ||||||
|  | 	/// Authors address
 | ||||||
| 	pub author: Address, | 	pub author: Address, | ||||||
| 	// TODO: get rid of this one
 | 	// TODO: get rid of this one
 | ||||||
|  | 	/// ?
 | ||||||
| 	pub miner: Address, | 	pub miner: Address, | ||||||
|  | 	/// State root hash
 | ||||||
| 	#[serde(rename="stateRoot")] | 	#[serde(rename="stateRoot")] | ||||||
| 	pub state_root: H256, | 	pub state_root: H256, | ||||||
|  | 	/// Transactions root hash
 | ||||||
| 	#[serde(rename="transactionsRoot")] | 	#[serde(rename="transactionsRoot")] | ||||||
| 	pub transactions_root: H256, | 	pub transactions_root: H256, | ||||||
|  | 	/// Transactions receipts root hash
 | ||||||
| 	#[serde(rename="receiptsRoot")] | 	#[serde(rename="receiptsRoot")] | ||||||
| 	pub receipts_root: H256, | 	pub receipts_root: H256, | ||||||
|  | 	/// Block number
 | ||||||
| 	pub number: OptionalValue<U256>, | 	pub number: OptionalValue<U256>, | ||||||
|  | 	/// Gas Used
 | ||||||
| 	#[serde(rename="gasUsed")] | 	#[serde(rename="gasUsed")] | ||||||
| 	pub gas_used: U256, | 	pub gas_used: U256, | ||||||
|  | 	/// Gas Limit
 | ||||||
| 	#[serde(rename="gasLimit")] | 	#[serde(rename="gasLimit")] | ||||||
| 	pub gas_limit: U256, | 	pub gas_limit: U256, | ||||||
|  | 	/// Extra data
 | ||||||
| 	#[serde(rename="extraData")] | 	#[serde(rename="extraData")] | ||||||
| 	pub extra_data: Bytes, | 	pub extra_data: Bytes, | ||||||
|  | 	/// Logs bloom
 | ||||||
| 	#[serde(rename="logsBloom")] | 	#[serde(rename="logsBloom")] | ||||||
| 	pub logs_bloom: H2048, | 	pub logs_bloom: H2048, | ||||||
|  | 	/// Timestamp
 | ||||||
| 	pub timestamp: U256, | 	pub timestamp: U256, | ||||||
|  | 	/// Difficulty
 | ||||||
| 	pub difficulty: U256, | 	pub difficulty: U256, | ||||||
|  | 	/// Total difficulty
 | ||||||
| 	#[serde(rename="totalDifficulty")] | 	#[serde(rename="totalDifficulty")] | ||||||
| 	pub total_difficulty: U256, | 	pub total_difficulty: U256, | ||||||
|  | 	/// Seal fields
 | ||||||
| 	#[serde(rename="sealFields")] | 	#[serde(rename="sealFields")] | ||||||
| 	pub seal_fields: Vec<Bytes>, | 	pub seal_fields: Vec<Bytes>, | ||||||
|  | 	/// Uncles' hashes
 | ||||||
| 	pub uncles: Vec<H256>, | 	pub uncles: Vec<H256>, | ||||||
|  | 	/// Transactions
 | ||||||
| 	pub transactions: BlockTransactions | 	pub transactions: BlockTransactions | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -21,9 +21,13 @@ use ethcore::client::BlockID; | |||||||
| /// Represents rpc api block number param.
 | /// Represents rpc api block number param.
 | ||||||
| #[derive(Debug, PartialEq, Clone)] | #[derive(Debug, PartialEq, Clone)] | ||||||
| pub enum BlockNumber { | pub enum BlockNumber { | ||||||
|  | 	/// Number
 | ||||||
| 	Num(u64), | 	Num(u64), | ||||||
|  | 	/// Latest block
 | ||||||
| 	Latest, | 	Latest, | ||||||
|  | 	/// Earliest block (genesis)
 | ||||||
| 	Earliest, | 	Earliest, | ||||||
|  | 	/// Pending block (being mined)
 | ||||||
| 	Pending | 	Pending | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -14,6 +14,8 @@ | |||||||
| // You should have received a copy of the GNU General Public License
 | // You should have received a copy of the GNU General Public License
 | ||||||
| // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
| 
 | 
 | ||||||
|  | //! Serializable wrapper around vector of bytes
 | ||||||
|  | 
 | ||||||
| use rustc_serialize::hex::ToHex; | use rustc_serialize::hex::ToHex; | ||||||
| use serde::{Serialize, Serializer, Deserialize, Deserializer, Error}; | use serde::{Serialize, Serializer, Deserialize, Deserializer, Error}; | ||||||
| use serde::de::Visitor; | use serde::de::Visitor; | ||||||
| @ -28,7 +30,10 @@ impl Bytes { | |||||||
| 	pub fn new(bytes: Vec<u8>) -> Bytes { | 	pub fn new(bytes: Vec<u8>) -> Bytes { | ||||||
| 		Bytes(bytes) | 		Bytes(bytes) | ||||||
| 	} | 	} | ||||||
| 	pub fn to_vec(self) -> Vec<u8> { self.0 } | 	/// Convert back to vector
 | ||||||
|  | 	pub fn to_vec(self) -> Vec<u8> { | ||||||
|  | 		self.0 | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Serialize for Bytes { | impl Serialize for Bytes { | ||||||
| @ -80,4 +85,3 @@ mod tests { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|  | |||||||
| @ -18,15 +18,23 @@ use util::hash::Address; | |||||||
| use util::numbers::U256; | use util::numbers::U256; | ||||||
| use v1::types::Bytes; | use v1::types::Bytes; | ||||||
| 
 | 
 | ||||||
|  | /// Call request
 | ||||||
| #[derive(Debug, Default, PartialEq, Deserialize)] | #[derive(Debug, Default, PartialEq, Deserialize)] | ||||||
| pub struct CallRequest { | pub struct CallRequest { | ||||||
|  | 	/// From
 | ||||||
| 	pub from: Option<Address>, | 	pub from: Option<Address>, | ||||||
|  | 	/// To
 | ||||||
| 	pub to: Option<Address>, | 	pub to: Option<Address>, | ||||||
|  | 	/// Gas Price
 | ||||||
| 	#[serde(rename="gasPrice")] | 	#[serde(rename="gasPrice")] | ||||||
| 	pub gas_price: Option<U256>, | 	pub gas_price: Option<U256>, | ||||||
|  | 	/// Gas
 | ||||||
| 	pub gas: Option<U256>, | 	pub gas: Option<U256>, | ||||||
|  | 	/// Value
 | ||||||
| 	pub value: Option<U256>, | 	pub value: Option<U256>, | ||||||
|  | 	/// Data
 | ||||||
| 	pub data: Option<Bytes>, | 	pub data: Option<Bytes>, | ||||||
|  | 	/// Nonce
 | ||||||
| 	pub nonce: Option<U256>, | 	pub nonce: Option<U256>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -22,10 +22,14 @@ use v1::types::BlockNumber; | |||||||
| use ethcore::filter::Filter as EthFilter; | use ethcore::filter::Filter as EthFilter; | ||||||
| use ethcore::client::BlockID; | use ethcore::client::BlockID; | ||||||
| 
 | 
 | ||||||
|  | /// Variadic value
 | ||||||
| #[derive(Debug, PartialEq, Clone)] | #[derive(Debug, PartialEq, Clone)] | ||||||
| pub enum VariadicValue<T> where T: Deserialize { | pub enum VariadicValue<T> where T: Deserialize { | ||||||
|  | 	/// Single
 | ||||||
| 	Single(T), | 	Single(T), | ||||||
|  | 	/// List
 | ||||||
| 	Multiple(Vec<T>), | 	Multiple(Vec<T>), | ||||||
|  | 	/// None
 | ||||||
| 	Null, | 	Null, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -44,17 +48,24 @@ impl<T> Deserialize for VariadicValue<T> where T: Deserialize { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Filter Address
 | ||||||
| pub type FilterAddress = VariadicValue<Address>; | pub type FilterAddress = VariadicValue<Address>; | ||||||
|  | /// Topic
 | ||||||
| pub type Topic = VariadicValue<H256>; | pub type Topic = VariadicValue<H256>; | ||||||
| 
 | 
 | ||||||
|  | /// Filter
 | ||||||
| #[derive(Debug, PartialEq, Clone, Deserialize)] | #[derive(Debug, PartialEq, Clone, Deserialize)] | ||||||
| #[serde(deny_unknown_fields)] | #[serde(deny_unknown_fields)] | ||||||
| pub struct Filter { | pub struct Filter { | ||||||
|  | 	/// From Block
 | ||||||
| 	#[serde(rename="fromBlock")] | 	#[serde(rename="fromBlock")] | ||||||
| 	pub from_block: Option<BlockNumber>, | 	pub from_block: Option<BlockNumber>, | ||||||
|  | 	/// To Block
 | ||||||
| 	#[serde(rename="toBlock")] | 	#[serde(rename="toBlock")] | ||||||
| 	pub to_block: Option<BlockNumber>, | 	pub to_block: Option<BlockNumber>, | ||||||
|  | 	/// Address
 | ||||||
| 	pub address: Option<FilterAddress>, | 	pub address: Option<FilterAddress>, | ||||||
|  | 	/// Topics
 | ||||||
| 	pub topics: Option<Vec<Topic>>, | 	pub topics: Option<Vec<Topic>>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -22,6 +22,7 @@ use serde::de::Visitor; | |||||||
| pub struct Index(usize); | pub struct Index(usize); | ||||||
| 
 | 
 | ||||||
| impl Index { | impl Index { | ||||||
|  | 	/// Convert to usize
 | ||||||
| 	pub fn value(&self) -> usize { | 	pub fn value(&self) -> usize { | ||||||
| 		self.0 | 		self.0 | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -18,21 +18,31 @@ use util::numbers::*; | |||||||
| use ethcore::log_entry::{LocalizedLogEntry, LogEntry}; | use ethcore::log_entry::{LocalizedLogEntry, LogEntry}; | ||||||
| use v1::types::Bytes; | use v1::types::Bytes; | ||||||
| 
 | 
 | ||||||
|  | /// Log
 | ||||||
| #[derive(Debug, Serialize, PartialEq, Eq, Hash, Clone)] | #[derive(Debug, Serialize, PartialEq, Eq, Hash, Clone)] | ||||||
| pub struct Log { | pub struct Log { | ||||||
|  | 	/// Address
 | ||||||
| 	pub address: Address, | 	pub address: Address, | ||||||
|  | 	/// Topics
 | ||||||
| 	pub topics: Vec<H256>, | 	pub topics: Vec<H256>, | ||||||
|  | 	/// Data
 | ||||||
| 	pub data: Bytes, | 	pub data: Bytes, | ||||||
|  | 	/// Block Hash
 | ||||||
| 	#[serde(rename="blockHash")] | 	#[serde(rename="blockHash")] | ||||||
| 	pub block_hash: Option<H256>, | 	pub block_hash: Option<H256>, | ||||||
|  | 	/// Block Number
 | ||||||
| 	#[serde(rename="blockNumber")] | 	#[serde(rename="blockNumber")] | ||||||
| 	pub block_number: Option<U256>, | 	pub block_number: Option<U256>, | ||||||
|  | 	/// Transaction Hash
 | ||||||
| 	#[serde(rename="transactionHash")] | 	#[serde(rename="transactionHash")] | ||||||
| 	pub transaction_hash: Option<H256>, | 	pub transaction_hash: Option<H256>, | ||||||
|  | 	/// Transaction Index
 | ||||||
| 	#[serde(rename="transactionIndex")] | 	#[serde(rename="transactionIndex")] | ||||||
| 	pub transaction_index: Option<U256>, | 	pub transaction_index: Option<U256>, | ||||||
|  | 	/// Log Index
 | ||||||
| 	#[serde(rename="logIndex")] | 	#[serde(rename="logIndex")] | ||||||
| 	pub log_index: Option<U256>, | 	pub log_index: Option<U256>, | ||||||
|  | 	/// Log Type
 | ||||||
| 	#[serde(rename="type")] | 	#[serde(rename="type")] | ||||||
| 	pub log_type: String, | 	pub log_type: String, | ||||||
| } | } | ||||||
|  | |||||||
| @ -14,6 +14,8 @@ | |||||||
| // You should have received a copy of the GNU General Public License
 | // You should have received a copy of the GNU General Public License
 | ||||||
| // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
| 
 | 
 | ||||||
|  | //! Structures used in RPC communication
 | ||||||
|  | 
 | ||||||
| #[cfg(feature = "serde_macros")] | #[cfg(feature = "serde_macros")] | ||||||
| include!("mod.rs.in"); | include!("mod.rs.in"); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -14,9 +14,9 @@ | |||||||
| // You should have received a copy of the GNU General Public License
 | // You should have received a copy of the GNU General Public License
 | ||||||
| // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
| 
 | 
 | ||||||
|  | mod bytes; | ||||||
| mod block; | mod block; | ||||||
| mod block_number; | mod block_number; | ||||||
| mod bytes; |  | ||||||
| mod filter; | mod filter; | ||||||
| mod index; | mod index; | ||||||
| mod log; | mod log; | ||||||
| @ -29,9 +29,9 @@ mod receipt; | |||||||
| mod trace; | mod trace; | ||||||
| mod trace_filter; | mod trace_filter; | ||||||
| 
 | 
 | ||||||
|  | pub use self::bytes::Bytes; | ||||||
| pub use self::block::{Block, BlockTransactions}; | pub use self::block::{Block, BlockTransactions}; | ||||||
| pub use self::block_number::BlockNumber; | pub use self::block_number::BlockNumber; | ||||||
| pub use self::bytes::Bytes; |  | ||||||
| pub use self::filter::Filter; | pub use self::filter::Filter; | ||||||
| pub use self::index::Index; | pub use self::index::Index; | ||||||
| pub use self::log::Log; | pub use self::log::Log; | ||||||
|  | |||||||
| @ -17,9 +17,12 @@ | |||||||
| use serde::{Serialize, Serializer}; | use serde::{Serialize, Serializer}; | ||||||
| use serde_json::Value; | use serde_json::Value; | ||||||
| 
 | 
 | ||||||
|  | /// Optional value
 | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub enum OptionalValue<T> where T: Serialize { | pub enum OptionalValue<T> where T: Serialize { | ||||||
|  | 	/// Some
 | ||||||
| 	Value(T), | 	Value(T), | ||||||
|  | 	/// None
 | ||||||
| 	Null | 	Null | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -19,22 +19,31 @@ use util::hash::{Address, H256}; | |||||||
| use v1::types::Log; | use v1::types::Log; | ||||||
| use ethcore::receipt::LocalizedReceipt; | use ethcore::receipt::LocalizedReceipt; | ||||||
| 
 | 
 | ||||||
|  | /// Receipt
 | ||||||
| #[derive(Debug, Serialize)] | #[derive(Debug, Serialize)] | ||||||
| pub struct Receipt { | pub struct Receipt { | ||||||
|  | 	/// Transaction Hash
 | ||||||
| 	#[serde(rename="transactionHash")] | 	#[serde(rename="transactionHash")] | ||||||
| 	pub transaction_hash: H256, | 	pub transaction_hash: H256, | ||||||
|  | 	/// Transaction index
 | ||||||
| 	#[serde(rename="transactionIndex")] | 	#[serde(rename="transactionIndex")] | ||||||
| 	pub transaction_index: U256, | 	pub transaction_index: U256, | ||||||
|  | 	/// Block hash
 | ||||||
| 	#[serde(rename="blockHash")] | 	#[serde(rename="blockHash")] | ||||||
| 	pub block_hash: H256, | 	pub block_hash: H256, | ||||||
|  | 	/// Block number
 | ||||||
| 	#[serde(rename="blockNumber")] | 	#[serde(rename="blockNumber")] | ||||||
| 	pub block_number: U256, | 	pub block_number: U256, | ||||||
|  | 	/// Cumulative gas used
 | ||||||
| 	#[serde(rename="cumulativeGasUsed")] | 	#[serde(rename="cumulativeGasUsed")] | ||||||
| 	pub cumulative_gas_used: U256, | 	pub cumulative_gas_used: U256, | ||||||
|  | 	/// Gas used
 | ||||||
| 	#[serde(rename="gasUsed")] | 	#[serde(rename="gasUsed")] | ||||||
| 	pub gas_used: U256, | 	pub gas_used: U256, | ||||||
|  | 	/// Contract address
 | ||||||
| 	#[serde(rename="contractAddress")] | 	#[serde(rename="contractAddress")] | ||||||
| 	pub contract_address: Option<Address>, | 	pub contract_address: Option<Address>, | ||||||
|  | 	/// Logs
 | ||||||
| 	pub logs: Vec<Log>, | 	pub logs: Vec<Log>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -17,19 +17,26 @@ | |||||||
| use serde::{Serialize, Serializer}; | use serde::{Serialize, Serializer}; | ||||||
| use util::numbers::*; | use util::numbers::*; | ||||||
| 
 | 
 | ||||||
|  | /// Sync info
 | ||||||
| #[derive(Default, Debug, Serialize, PartialEq)] | #[derive(Default, Debug, Serialize, PartialEq)] | ||||||
| pub struct SyncInfo { | pub struct SyncInfo { | ||||||
|  | 	/// Starting block
 | ||||||
| 	#[serde(rename="startingBlock")] | 	#[serde(rename="startingBlock")] | ||||||
| 	pub starting_block: U256, | 	pub starting_block: U256, | ||||||
|  | 	/// Current block
 | ||||||
| 	#[serde(rename="currentBlock")] | 	#[serde(rename="currentBlock")] | ||||||
| 	pub current_block: U256, | 	pub current_block: U256, | ||||||
|  | 	/// Highest block seen so far
 | ||||||
| 	#[serde(rename="highestBlock")] | 	#[serde(rename="highestBlock")] | ||||||
| 	pub highest_block: U256, | 	pub highest_block: U256, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Sync status
 | ||||||
| #[derive(Debug, PartialEq)] | #[derive(Debug, PartialEq)] | ||||||
| pub enum SyncStatus { | pub enum SyncStatus { | ||||||
|  | 	/// Info when syncing
 | ||||||
| 	Info(SyncInfo), | 	Info(SyncInfo), | ||||||
|  | 	/// Not syncing
 | ||||||
| 	None | 	None | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -19,11 +19,16 @@ use ethcore::trace::trace; | |||||||
| use ethcore::trace::LocalizedTrace; | use ethcore::trace::LocalizedTrace; | ||||||
| use v1::types::Bytes; | use v1::types::Bytes; | ||||||
| 
 | 
 | ||||||
|  | /// Create response
 | ||||||
| #[derive(Debug, Serialize)] | #[derive(Debug, Serialize)] | ||||||
| pub struct Create { | pub struct Create { | ||||||
|  | 	/// Sender
 | ||||||
| 	from: Address, | 	from: Address, | ||||||
|  | 	/// Value
 | ||||||
| 	value: U256, | 	value: U256, | ||||||
|  | 	/// Gas
 | ||||||
| 	gas: U256, | 	gas: U256, | ||||||
|  | 	/// Initialization code
 | ||||||
| 	init: Bytes, | 	init: Bytes, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -38,12 +43,18 @@ impl From<trace::Create> for Create { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Call response
 | ||||||
| #[derive(Debug, Serialize)] | #[derive(Debug, Serialize)] | ||||||
| pub struct Call { | pub struct Call { | ||||||
|  | 	/// Sender
 | ||||||
| 	from: Address, | 	from: Address, | ||||||
|  | 	/// Recipient
 | ||||||
| 	to: Address, | 	to: Address, | ||||||
|  | 	/// Transfered Value
 | ||||||
| 	value: U256, | 	value: U256, | ||||||
|  | 	/// Gas
 | ||||||
| 	gas: U256, | 	gas: U256, | ||||||
|  | 	/// Input data
 | ||||||
| 	input: Bytes, | 	input: Bytes, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -59,10 +70,13 @@ impl From<trace::Call> for Call { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Action
 | ||||||
| #[derive(Debug, Serialize)] | #[derive(Debug, Serialize)] | ||||||
| pub enum Action { | pub enum Action { | ||||||
|  | 	/// Call
 | ||||||
| 	#[serde(rename="call")] | 	#[serde(rename="call")] | ||||||
| 	Call(Call), | 	Call(Call), | ||||||
|  | 	/// Create
 | ||||||
| 	#[serde(rename="create")] | 	#[serde(rename="create")] | ||||||
| 	Create(Create), | 	Create(Create), | ||||||
| } | } | ||||||
| @ -76,10 +90,13 @@ impl From<trace::Action> for Action { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Call Result
 | ||||||
| #[derive(Debug, Serialize)] | #[derive(Debug, Serialize)] | ||||||
| pub struct CallResult { | pub struct CallResult { | ||||||
|  | 	/// Gas used
 | ||||||
| 	#[serde(rename="gasUsed")] | 	#[serde(rename="gasUsed")] | ||||||
| 	gas_used: U256, | 	gas_used: U256, | ||||||
|  | 	/// Output bytes
 | ||||||
| 	output: Bytes, | 	output: Bytes, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -92,11 +109,15 @@ impl From<trace::CallResult> for CallResult { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Craete Result
 | ||||||
| #[derive(Debug, Serialize)] | #[derive(Debug, Serialize)] | ||||||
| pub struct CreateResult { | pub struct CreateResult { | ||||||
|  | 	/// Gas used
 | ||||||
| 	#[serde(rename="gasUsed")] | 	#[serde(rename="gasUsed")] | ||||||
| 	gas_used: U256, | 	gas_used: U256, | ||||||
|  | 	/// Code
 | ||||||
| 	code: Bytes, | 	code: Bytes, | ||||||
|  | 	/// Assigned address
 | ||||||
| 	address: Address, | 	address: Address, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -110,14 +131,19 @@ impl From<trace::CreateResult> for CreateResult { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Response
 | ||||||
| #[derive(Debug, Serialize)] | #[derive(Debug, Serialize)] | ||||||
| pub enum Res { | pub enum Res { | ||||||
|  | 	/// Call
 | ||||||
| 	#[serde(rename="call")] | 	#[serde(rename="call")] | ||||||
| 	Call(CallResult), | 	Call(CallResult), | ||||||
|  | 	/// Create
 | ||||||
| 	#[serde(rename="create")] | 	#[serde(rename="create")] | ||||||
| 	Create(CreateResult), | 	Create(CreateResult), | ||||||
|  | 	/// Call failure
 | ||||||
| 	#[serde(rename="failedCall")] | 	#[serde(rename="failedCall")] | ||||||
| 	FailedCall, | 	FailedCall, | ||||||
|  | 	/// Creation failure
 | ||||||
| 	#[serde(rename="failedCreate")] | 	#[serde(rename="failedCreate")] | ||||||
| 	FailedCreate, | 	FailedCreate, | ||||||
| } | } | ||||||
| @ -133,19 +159,28 @@ impl From<trace::Res> for Res { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Trace
 | ||||||
| #[derive(Debug, Serialize)] | #[derive(Debug, Serialize)] | ||||||
| pub struct Trace { | pub struct Trace { | ||||||
|  | 	/// Action
 | ||||||
| 	action: Action, | 	action: Action, | ||||||
|  | 	/// Result
 | ||||||
| 	result: Res, | 	result: Res, | ||||||
|  | 	/// Trace address
 | ||||||
| 	#[serde(rename="traceAddress")] | 	#[serde(rename="traceAddress")] | ||||||
| 	trace_address: Vec<U256>, | 	trace_address: Vec<U256>, | ||||||
|  | 	/// Subtraces
 | ||||||
| 	subtraces: U256, | 	subtraces: U256, | ||||||
|  | 	/// Transaction position
 | ||||||
| 	#[serde(rename="transactionPosition")] | 	#[serde(rename="transactionPosition")] | ||||||
| 	transaction_position: U256, | 	transaction_position: U256, | ||||||
|  | 	/// Transaction hash
 | ||||||
| 	#[serde(rename="transactionHash")] | 	#[serde(rename="transactionHash")] | ||||||
| 	transaction_hash: H256, | 	transaction_hash: H256, | ||||||
|  | 	/// Block Number
 | ||||||
| 	#[serde(rename="blockNumber")] | 	#[serde(rename="blockNumber")] | ||||||
| 	block_number: U256, | 	block_number: U256, | ||||||
|  | 	/// Block Hash
 | ||||||
| 	#[serde(rename="blockHash")] | 	#[serde(rename="blockHash")] | ||||||
| 	block_hash: H256, | 	block_hash: H256, | ||||||
| } | } | ||||||
|  | |||||||
| @ -21,14 +21,19 @@ use ethcore::client::BlockID; | |||||||
| use ethcore::client; | use ethcore::client; | ||||||
| use super::BlockNumber; | use super::BlockNumber; | ||||||
| 
 | 
 | ||||||
|  | /// Trace filter
 | ||||||
| #[derive(Debug, PartialEq, Deserialize)] | #[derive(Debug, PartialEq, Deserialize)] | ||||||
| pub struct TraceFilter { | pub struct TraceFilter { | ||||||
|  | 	/// From block
 | ||||||
| 	#[serde(rename="fromBlock")] | 	#[serde(rename="fromBlock")] | ||||||
| 	pub from_block: Option<BlockNumber>, | 	pub from_block: Option<BlockNumber>, | ||||||
|  | 	/// To block
 | ||||||
| 	#[serde(rename="toBlock")] | 	#[serde(rename="toBlock")] | ||||||
| 	pub to_block: Option<BlockNumber>, | 	pub to_block: Option<BlockNumber>, | ||||||
|  | 	/// From address
 | ||||||
| 	#[serde(rename="fromAddress")] | 	#[serde(rename="fromAddress")] | ||||||
| 	pub from_address: Option<Vec<Address>>, | 	pub from_address: Option<Vec<Address>>, | ||||||
|  | 	/// To address
 | ||||||
| 	#[serde(rename="toAddress")] | 	#[serde(rename="toAddress")] | ||||||
| 	pub to_address: Option<Vec<Address>>, | 	pub to_address: Option<Vec<Address>>, | ||||||
| } | } | ||||||
|  | |||||||
| @ -18,22 +18,34 @@ use util::numbers::*; | |||||||
| use ethcore::transaction::{LocalizedTransaction, Action, SignedTransaction}; | use ethcore::transaction::{LocalizedTransaction, Action, SignedTransaction}; | ||||||
| use v1::types::{Bytes, OptionalValue}; | use v1::types::{Bytes, OptionalValue}; | ||||||
| 
 | 
 | ||||||
|  | /// Transaction
 | ||||||
| #[derive(Debug, Default, Serialize)] | #[derive(Debug, Default, Serialize)] | ||||||
| pub struct Transaction { | pub struct Transaction { | ||||||
|  | 	/// Hash
 | ||||||
| 	pub hash: H256, | 	pub hash: H256, | ||||||
|  | 	/// Nonce
 | ||||||
| 	pub nonce: U256, | 	pub nonce: U256, | ||||||
|  | 	/// Block hash
 | ||||||
| 	#[serde(rename="blockHash")] | 	#[serde(rename="blockHash")] | ||||||
| 	pub block_hash: OptionalValue<H256>, | 	pub block_hash: OptionalValue<H256>, | ||||||
|  | 	/// Block number
 | ||||||
| 	#[serde(rename="blockNumber")] | 	#[serde(rename="blockNumber")] | ||||||
| 	pub block_number: OptionalValue<U256>, | 	pub block_number: OptionalValue<U256>, | ||||||
|  | 	/// Transaction Index
 | ||||||
| 	#[serde(rename="transactionIndex")] | 	#[serde(rename="transactionIndex")] | ||||||
| 	pub transaction_index: OptionalValue<U256>, | 	pub transaction_index: OptionalValue<U256>, | ||||||
|  | 	/// Sender
 | ||||||
| 	pub from: Address, | 	pub from: Address, | ||||||
|  | 	/// Recipient
 | ||||||
| 	pub to: OptionalValue<Address>, | 	pub to: OptionalValue<Address>, | ||||||
|  | 	/// Transfered value
 | ||||||
| 	pub value: U256, | 	pub value: U256, | ||||||
|  | 	/// Gas Price
 | ||||||
| 	#[serde(rename="gasPrice")] | 	#[serde(rename="gasPrice")] | ||||||
| 	pub gas_price: U256, | 	pub gas_price: U256, | ||||||
|  | 	/// Gas
 | ||||||
| 	pub gas: U256, | 	pub gas: U256, | ||||||
|  | 	/// Data
 | ||||||
| 	pub input: Bytes | 	pub input: Bytes | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -14,19 +14,29 @@ | |||||||
| // You should have received a copy of the GNU General Public License
 | // You should have received a copy of the GNU General Public License
 | ||||||
| // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
| 
 | 
 | ||||||
|  | //! `TransactionRequest` type
 | ||||||
|  | 
 | ||||||
| use util::hash::Address; | use util::hash::Address; | ||||||
| use util::numbers::U256; | use util::numbers::U256; | ||||||
| use v1::types::Bytes; | use v1::types::bytes::Bytes; | ||||||
| 
 | 
 | ||||||
| #[derive(Debug, Default, PartialEq, Deserialize)] | /// Transaction request coming from RPC
 | ||||||
|  | #[derive(Debug, Clone, Default, Eq, PartialEq, Hash, Deserialize)] | ||||||
| pub struct TransactionRequest { | pub struct TransactionRequest { | ||||||
|  | 	/// Sender
 | ||||||
| 	pub from: Address, | 	pub from: Address, | ||||||
|  | 	/// Recipient
 | ||||||
| 	pub to: Option<Address>, | 	pub to: Option<Address>, | ||||||
|  | 	/// Gas Price
 | ||||||
| 	#[serde(rename="gasPrice")] | 	#[serde(rename="gasPrice")] | ||||||
| 	pub gas_price: Option<U256>, | 	pub gas_price: Option<U256>, | ||||||
|  | 	/// Gas
 | ||||||
| 	pub gas: Option<U256>, | 	pub gas: Option<U256>, | ||||||
|  | 	/// Value of transaction in wei
 | ||||||
| 	pub value: Option<U256>, | 	pub value: Option<U256>, | ||||||
|  | 	/// Additional data sent with transaction
 | ||||||
| 	pub data: Option<Bytes>, | 	pub data: Option<Bytes>, | ||||||
|  | 	/// Transaction's nonce
 | ||||||
| 	pub nonce: Option<U256>, | 	pub nonce: Option<U256>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -37,7 +47,7 @@ mod tests { | |||||||
| 	use serde_json; | 	use serde_json; | ||||||
| 	use util::numbers::{U256}; | 	use util::numbers::{U256}; | ||||||
| 	use util::hash::Address; | 	use util::hash::Address; | ||||||
| 	use v1::types::Bytes; | 	use v1::types::bytes::Bytes; | ||||||
| 	use super::*; | 	use super::*; | ||||||
| 
 | 
 | ||||||
| 	#[test] | 	#[test] | ||||||
| @ -126,3 +136,4 @@ mod tests { | |||||||
| 		}); | 		}); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | |||||||
| @ -9,13 +9,23 @@ build = "build.rs" | |||||||
| 
 | 
 | ||||||
| [build-dependencies] | [build-dependencies] | ||||||
| rustc_version = "0.1" | rustc_version = "0.1" | ||||||
|  | serde_codegen = { version = "0.7.0", optional = true } | ||||||
|  | syntex = "^0.32.0" | ||||||
| 
 | 
 | ||||||
| [dependencies] | [dependencies] | ||||||
| ethcore-util = { path = "../util" } | serde = "0.7.0" | ||||||
|  | serde_json = "0.7.0" | ||||||
|  | rustc-serialize = "0.3" | ||||||
| log = "0.3" | log = "0.3" | ||||||
| env_logger = "0.3" | env_logger = "0.3" | ||||||
|  | ws = "0.4.7" | ||||||
|  | ethcore-util = { path = "../util" } | ||||||
|  | ethcore-rpc = { path = "../rpc" } | ||||||
|  | 
 | ||||||
|  | serde_macros = { version = "0.7.0", optional = true } | ||||||
| clippy = { version = "0.0.69", optional = true} | clippy = { version = "0.0.69", optional = true} | ||||||
| 
 | 
 | ||||||
| [features] | [features] | ||||||
| default = [] | default = ["serde_codegen"] | ||||||
|  | nightly = ["serde_macros"] | ||||||
| dev = ["clippy"] | dev = ["clippy"] | ||||||
|  | |||||||
| @ -19,7 +19,34 @@ extern crate rustc_version; | |||||||
| use rustc_version::{version_meta, Channel}; | use rustc_version::{version_meta, Channel}; | ||||||
| 
 | 
 | ||||||
| fn main() { | fn main() { | ||||||
|  | 	serde::main(); | ||||||
| 	if let Channel::Nightly = version_meta().channel { | 	if let Channel::Nightly = version_meta().channel { | ||||||
| 		println!("cargo:rustc-cfg=nightly"); | 		println!("cargo:rustc-cfg=nightly"); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | #[cfg(not(feature = "serde_macros"))] | ||||||
|  | mod serde { | ||||||
|  |     extern crate syntex; | ||||||
|  |     extern crate serde_codegen; | ||||||
|  | 
 | ||||||
|  |     use std::env; | ||||||
|  |     use std::path::Path; | ||||||
|  | 
 | ||||||
|  |     pub fn main() { | ||||||
|  |         let out_dir = env::var_os("OUT_DIR").unwrap(); | ||||||
|  | 
 | ||||||
|  |         let src = Path::new("src/types/mod.rs.in"); | ||||||
|  |         let dst = Path::new(&out_dir).join("mod.rs"); | ||||||
|  | 
 | ||||||
|  |         let mut registry = syntex::Registry::new(); | ||||||
|  | 
 | ||||||
|  |         serde_codegen::register(&mut registry); | ||||||
|  |         registry.expand("", &src, &dst).unwrap(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(feature = "serde_macros")] | ||||||
|  | mod serde { | ||||||
|  |     pub fn main() {} | ||||||
|  | } | ||||||
|  | |||||||
| @ -17,6 +17,8 @@ | |||||||
| #![warn(missing_docs)] | #![warn(missing_docs)] | ||||||
| #![cfg_attr(all(nightly, feature="dev"), feature(plugin))] | #![cfg_attr(all(nightly, feature="dev"), feature(plugin))] | ||||||
| #![cfg_attr(all(nightly, feature="dev"), plugin(clippy))] | #![cfg_attr(all(nightly, feature="dev"), plugin(clippy))] | ||||||
|  | // Generated by serde
 | ||||||
|  | #![cfg_attr(all(nightly, feature="dev"), allow(redundant_closure_call))] | ||||||
| 
 | 
 | ||||||
| //! Signer module
 | //! Signer module
 | ||||||
| //!
 | //!
 | ||||||
| @ -28,12 +30,33 @@ | |||||||
| //! and their responsibility is to confirm (or confirm and sign)
 | //! and their responsibility is to confirm (or confirm and sign)
 | ||||||
| //! the transaction for you.
 | //! the transaction for you.
 | ||||||
| //!
 | //!
 | ||||||
|  | //! ```
 | ||||||
|  | //! extern crate ethcore_signer;
 | ||||||
| //!
 | //!
 | ||||||
|  | //! use ethcore_signer::Server;
 | ||||||
|  | //!
 | ||||||
|  | //!	fn main() {
 | ||||||
|  | //!	 let _server = Server::start("127.0.0.1:8084".parse().unwrap());
 | ||||||
|  | //!	}
 | ||||||
|  | //! ```
 | ||||||
| 
 | 
 | ||||||
| #[macro_use] | #[macro_use] | ||||||
| extern crate log; | extern crate log; | ||||||
| extern crate env_logger; | extern crate env_logger; | ||||||
| 
 | 
 | ||||||
|  | extern crate serde; | ||||||
|  | extern crate serde_json; | ||||||
|  | extern crate rustc_serialize; | ||||||
|  | 
 | ||||||
|  | extern crate ethcore_util as util; | ||||||
|  | extern crate ethcore_rpc as rpc; | ||||||
|  | extern crate ws; | ||||||
|  | 
 | ||||||
|  | mod signing_queue; | ||||||
|  | mod ws_server; | ||||||
|  | 
 | ||||||
|  | pub use ws_server::*; | ||||||
|  | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod tests { | mod tests { | ||||||
| 	#[test] | 	#[test] | ||||||
|  | |||||||
							
								
								
									
										74
									
								
								signer/src/signing_queue.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								signer/src/signing_queue.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,74 @@ | |||||||
|  | // Copyright 2015, 2016 Ethcore (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 std::collections::HashSet; | ||||||
|  | use rpc::v1::types::TransactionRequest; | ||||||
|  | 
 | ||||||
|  | pub trait SigningQueue { | ||||||
|  | 	fn add_request(&mut self, transaction: TransactionRequest); | ||||||
|  | 
 | ||||||
|  | 	fn remove_request(&mut self, id: TransactionRequest); | ||||||
|  | 
 | ||||||
|  | 	fn requests(&self) -> &HashSet<TransactionRequest>; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl SigningQueue for HashSet<TransactionRequest> { | ||||||
|  | 	fn add_request(&mut self, transaction: TransactionRequest) { | ||||||
|  | 		self.insert(transaction); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn remove_request(&mut self, id: TransactionRequest) { | ||||||
|  | 		self.remove(&id); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn requests(&self) -> &HashSet<TransactionRequest> { | ||||||
|  | 		self | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #[cfg(test)] | ||||||
|  | mod test { | ||||||
|  | 	use std::collections::HashSet; | ||||||
|  | 	use util::hash::Address; | ||||||
|  | 	use util::numbers::U256; | ||||||
|  | 	use rpc::v1::types::TransactionRequest; | ||||||
|  | 	use super::*; | ||||||
|  | 
 | ||||||
|  | 	#[test] | ||||||
|  | 	fn should_work_for_hashset() { | ||||||
|  | 		// given
 | ||||||
|  | 		let mut queue = HashSet::new(); | ||||||
|  | 
 | ||||||
|  | 		let request = TransactionRequest { | ||||||
|  | 			from: Address::from(1), | ||||||
|  | 			to: Some(Address::from(2)), | ||||||
|  | 			gas_price: None, | ||||||
|  | 			gas: None, | ||||||
|  | 			value: Some(U256::from(10_000_000)), | ||||||
|  | 			data: None, | ||||||
|  | 			nonce: None, | ||||||
|  | 		}; | ||||||
|  | 
 | ||||||
|  | 		// when
 | ||||||
|  | 		queue.add_request(request.clone()); | ||||||
|  | 		let all = queue.requests(); | ||||||
|  | 
 | ||||||
|  | 		// then
 | ||||||
|  | 		assert_eq!(all.len(), 1); | ||||||
|  | 		assert!(all.contains(&request)); | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										23
									
								
								signer/src/types/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								signer/src/types/mod.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,23 @@ | |||||||
|  | // Copyright 2015, 2016 Ethcore (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/>.
 | ||||||
|  | 
 | ||||||
|  | //! Reusable types with JSON Serialization.
 | ||||||
|  | 
 | ||||||
|  | #[cfg(feature = "serde_macros")] | ||||||
|  | include!("mod.rs.in"); | ||||||
|  | 
 | ||||||
|  | #[cfg(not(feature = "serde_macros"))] | ||||||
|  | include!(concat!(env!("OUT_DIR"), "/mod.rs")); | ||||||
							
								
								
									
										25
									
								
								signer/src/types/mod.rs.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								signer/src/types/mod.rs.in
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,25 @@ | |||||||
|  | // Copyright 2015, 2016 Ethcore (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/>.
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // TODO [ToDr] Types are empty for now. But they are about to come.
 | ||||||
							
								
								
									
										128
									
								
								signer/src/ws_server.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								signer/src/ws_server.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,128 @@ | |||||||
|  | // Copyright 2015, 2016 Ethcore (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/>.
 | ||||||
|  | 
 | ||||||
|  | //! `WebSockets` server.
 | ||||||
|  | 
 | ||||||
|  | use ws; | ||||||
|  | use std; | ||||||
|  | use std::thread; | ||||||
|  | use std::ops::Drop; | ||||||
|  | use std::sync::Arc; | ||||||
|  | use std::sync::atomic::{AtomicUsize, Ordering}; | ||||||
|  | use std::net::SocketAddr; | ||||||
|  | use util::panics::{PanicHandler, OnPanicListener, MayPanic}; | ||||||
|  | 
 | ||||||
|  | /// Signer startup error
 | ||||||
|  | #[derive(Debug)] | ||||||
|  | pub enum ServerError { | ||||||
|  | 	/// Wrapped `std::io::Error`
 | ||||||
|  | 	IoError(std::io::Error), | ||||||
|  | 	/// Other `ws-rs` error
 | ||||||
|  | 	WebSocket(ws::Error) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<ws::Error> for ServerError { | ||||||
|  | 	fn from(err: ws::Error) -> Self { | ||||||
|  | 		match err.kind { | ||||||
|  | 			ws::ErrorKind::Io(e) => ServerError::IoError(e), | ||||||
|  | 			_ => ServerError::WebSocket(err), | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// `WebSockets` server implementation.
 | ||||||
|  | pub struct Server { | ||||||
|  | 	handle: Option<thread::JoinHandle<ws::WebSocket<Factory>>>, | ||||||
|  | 	broadcaster: ws::Sender, | ||||||
|  | 	panic_handler: Arc<PanicHandler>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Server { | ||||||
|  | 	/// Starts a new `WebSocket` server in separate thread.
 | ||||||
|  | 	/// Returns a `Server` handle which closes the server when droped.
 | ||||||
|  | 	pub fn start(addr: SocketAddr) -> Result<Server, ServerError> { | ||||||
|  | 		let config = { | ||||||
|  | 			let mut config = ws::Settings::default(); | ||||||
|  | 			config.max_connections = 5; | ||||||
|  | 			config.method_strict = true; | ||||||
|  | 			config | ||||||
|  | 		}; | ||||||
|  | 
 | ||||||
|  | 		// Create WebSocket
 | ||||||
|  | 		let session_id = Arc::new(AtomicUsize::new(1)); | ||||||
|  | 		let ws = try!(ws::Builder::new().with_settings(config).build(Factory { | ||||||
|  | 			session_id: session_id, | ||||||
|  | 		})); | ||||||
|  | 
 | ||||||
|  | 		let panic_handler = PanicHandler::new_in_arc(); | ||||||
|  | 		let ph = panic_handler.clone(); | ||||||
|  | 		let broadcaster = ws.broadcaster(); | ||||||
|  | 		// Spawn a thread with event loop
 | ||||||
|  | 		let handle = thread::spawn(move || { | ||||||
|  | 			ph.catch_panic(move || { | ||||||
|  | 				ws.listen(addr).unwrap() | ||||||
|  | 			}).unwrap() | ||||||
|  | 		}); | ||||||
|  | 
 | ||||||
|  | 		// Return a handle
 | ||||||
|  | 		Ok(Server { | ||||||
|  | 			handle: Some(handle), | ||||||
|  | 			broadcaster: broadcaster, | ||||||
|  | 			panic_handler: panic_handler, | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl MayPanic for Server { | ||||||
|  | 	fn on_panic<F>(&self, closure: F) where F: OnPanicListener { | ||||||
|  | 		self.panic_handler.on_panic(closure); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Drop for Server { | ||||||
|  | 	fn drop(&mut self) { | ||||||
|  | 		self.broadcaster.shutdown().expect("WsServer should close nicely."); | ||||||
|  | 		self.handle.take().unwrap().join().unwrap(); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct Session { | ||||||
|  | 	id: usize, | ||||||
|  | 	out: ws::Sender, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl ws::Handler for Session { | ||||||
|  | 	fn on_open(&mut self, _shake: ws::Handshake) -> ws::Result<()> { | ||||||
|  | 		try!(self.out.send(format!("Hello client no: {}. We are not implemented yet.", self.id))); | ||||||
|  | 		try!(self.out.close(ws::CloseCode::Normal)); | ||||||
|  | 		Ok(()) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct Factory { | ||||||
|  | 	session_id: Arc<AtomicUsize>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl ws::Factory for Factory { | ||||||
|  | 	type Handler = Session; | ||||||
|  | 
 | ||||||
|  | 	fn connection_made(&mut self, sender: ws::Sender) -> Self::Handler { | ||||||
|  | 		Session { | ||||||
|  | 			id: self.session_id.fetch_add(1, Ordering::SeqCst), | ||||||
|  | 			out: sender, | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user