2016-04-21 16:45:04 +02:00
// 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 ::env ;
use std ::fs ::File ;
2016-06-29 16:26:19 +02:00
use std ::time ::Duration ;
2016-04-21 16:45:04 +02:00
use std ::io ::{ BufRead , BufReader } ;
use std ::net ::{ SocketAddr , IpAddr } ;
use std ::path ::PathBuf ;
use cli ::{ USAGE , Args } ;
use docopt ::Docopt ;
use die ::* ;
use util ::* ;
2016-06-29 17:50:27 +02:00
use util ::log ::Colour ::* ;
2016-06-20 00:10:34 +02:00
use ethcore ::account_provider ::AccountProvider ;
2016-04-21 19:19:42 +02:00
use util ::network_settings ::NetworkSettings ;
2016-07-05 17:50:46 +02:00
use ethcore ::client ::{ append_path , get_db_path , Mode , ClientConfig , DatabaseCompactionProfile , Switch , VMType } ;
2016-07-08 17:26:06 +02:00
use ethcore ::miner ::{ MinerOptions , PendingSet , GasPricer , GasPriceCalibratorOptions } ;
2016-04-21 16:45:04 +02:00
use ethcore ::ethereum ;
use ethcore ::spec ::Spec ;
use ethsync ::SyncConfig ;
2016-05-04 15:37:09 +02:00
use rpc ::IpcConfiguration ;
2016-04-21 16:45:04 +02:00
pub struct Configuration {
pub args : Args
}
2016-05-13 17:32:32 +02:00
pub struct Directories {
pub keys : String ,
pub db : String ,
2016-06-03 11:51:11 +02:00
pub dapps : String ,
2016-06-07 17:21:19 +02:00
pub signer : String ,
2016-05-13 17:32:32 +02:00
}
2016-04-21 16:45:04 +02:00
impl Configuration {
pub fn parse ( ) -> Self {
Configuration {
args : Docopt ::new ( USAGE ) . and_then ( | d | d . decode ( ) ) . unwrap_or_else ( | e | e . exit ( ) ) ,
}
}
2016-07-05 17:50:46 +02:00
pub fn mode ( & self ) -> Mode {
match & ( self . args . flag_mode [ .. ] ) {
" active " = > Mode ::Active ,
" passive " = > Mode ::Passive ( Duration ::from_secs ( self . args . flag_mode_timeout ) , Duration ::from_secs ( self . args . flag_mode_alarm ) ) ,
" dark " = > Mode ::Dark ( Duration ::from_secs ( self . args . flag_mode_timeout ) ) ,
_ = > die! ( " {}: Invalid address for --mode. Must be one of active, passive or dark. " , self . args . flag_mode ) ,
}
}
2016-04-30 19:58:28 +02:00
fn net_port ( & self ) -> u16 {
self . args . flag_port
}
fn chain ( & self ) -> String {
if self . args . flag_testnet {
" morden " . to_owned ( )
} else {
self . args . flag_chain . clone ( )
}
}
fn max_peers ( & self ) -> u32 {
self . args . flag_maxpeers . unwrap_or ( self . args . flag_peers ) as u32
}
2016-06-27 18:27:06 +02:00
fn decode_u256 ( d : & str , argument : & str ) -> U256 {
U256 ::from_dec_str ( d ) . unwrap_or_else ( | _ |
U256 ::from_str ( clean_0x ( d ) ) . unwrap_or_else ( | _ |
die! ( " {}: Invalid numeric value for {}. Must be either a decimal or a hex number. " , d , argument )
)
)
}
2016-06-29 20:07:21 +02:00
fn work_notify ( & self ) -> Vec < String > {
self . args . flag_notify_work . as_ref ( ) . map_or_else ( Vec ::new , | s | s . split ( ',' ) . map ( | s | s . to_owned ( ) ) . collect ( ) )
2016-06-29 15:37:11 +02:00
}
2016-06-27 17:23:54 +02:00
pub fn miner_options ( & self ) -> MinerOptions {
let ( own , ext ) = match self . args . flag_reseal_on_txs . as_str ( ) {
" none " = > ( false , false ) ,
" own " = > ( true , false ) ,
" ext " = > ( false , true ) ,
" all " = > ( true , true ) ,
x = > die! ( " {}: Invalid value for --reseal option. Use --help for more information. " , x )
} ;
MinerOptions {
2016-06-29 15:37:11 +02:00
new_work_notify : self . work_notify ( ) ,
2016-06-27 17:23:54 +02:00
force_sealing : self . args . flag_force_sealing ,
reseal_on_external_tx : ext ,
reseal_on_own_tx : own ,
2016-06-28 10:40:35 +02:00
tx_gas_limit : self . args . flag_tx_gas_limit . as_ref ( ) . map_or ( ! U256 ::zero ( ) , | d | Self ::decode_u256 ( d , " --tx-gas-limit " ) ) ,
2016-06-27 20:19:01 +02:00
tx_queue_size : self . args . flag_tx_queue_size ,
2016-06-27 19:06:54 +02:00
pending_set : match self . args . flag_relay_set . as_str ( ) {
" cheap " = > PendingSet ::AlwaysQueue ,
" strict " = > PendingSet ::AlwaysSealing ,
" lenient " = > PendingSet ::SealingOrElseQueue ,
x = > die! ( " {}: Invalid value for --relay-set option. Use --help for more information. " , x )
2016-06-27 18:27:06 +02:00
} ,
2016-06-29 16:26:19 +02:00
reseal_min_period : Duration ::from_millis ( self . args . flag_reseal_min_period ) ,
work_queue_size : self . args . flag_work_queue_size ,
2016-06-30 12:56:58 +02:00
enable_resubmission : ! self . args . flag_remove_solved ,
2016-06-27 17:23:54 +02:00
}
}
2016-06-26 22:02:17 +02:00
pub fn author ( & self ) -> Option < Address > {
self . args . flag_etherbase . as_ref ( )
. or ( self . args . flag_author . as_ref ( ) )
. map ( | d | Address ::from_str ( clean_0x ( d ) ) . unwrap_or_else ( | _ | {
die! ( " {}: Invalid address for --author. Must be 40 hex characters, with or without the 0x at the beginning. " , d )
} ) )
2016-04-21 16:45:04 +02:00
}
pub fn gas_floor_target ( & self ) -> U256 {
2016-07-05 18:18:35 +02:00
let d = & self . args . flag_gas_floor_target ;
U256 ::from_dec_str ( d ) . unwrap_or_else ( | _ | {
die! ( " {}: Invalid target gas floor given. Must be a decimal unsigned 256-bit number. " , d )
} )
2016-04-21 16:45:04 +02:00
}
2016-06-23 14:29:16 +02:00
pub fn gas_ceil_target ( & self ) -> U256 {
2016-07-05 18:18:35 +02:00
let d = & self . args . flag_gas_cap ;
U256 ::from_dec_str ( d ) . unwrap_or_else ( | _ | {
die! ( " {}: Invalid target gas ceiling given. Must be a decimal unsigned 256-bit number. " , d )
} )
2016-06-23 14:29:16 +02:00
}
2016-06-19 13:20:14 +02:00
2016-07-08 17:26:06 +02:00
fn to_duration ( s : & str ) -> Duration {
let bad = | _ | {
die! ( " {}: Invalid duration given. See parity --help for more information. " , s )
} ;
Duration ::from_secs ( match s {
" twice-daily " = > 12 * 60 * 60 ,
" half-hourly " = > 30 * 60 ,
" 1second " | " 1 second " | " second " = > 1 ,
" 1minute " | " 1 minute " | " minute " = > 60 ,
2016-07-10 13:18:33 +02:00
" hourly " | " 1hour " | " 1 hour " | " hour " = > 60 * 60 ,
" daily " | " 1day " | " 1 day " | " day " = > 24 * 60 * 60 ,
2016-07-08 17:26:06 +02:00
x if x . ends_with ( " seconds " ) = > FromStr ::from_str ( & x [ 0 .. x . len ( ) - 7 ] ) . unwrap_or_else ( bad ) ,
x if x . ends_with ( " minutes " ) = > FromStr ::from_str ( & x [ 0 .. x . len ( ) - 7 ] ) . unwrap_or_else ( bad ) * 60 ,
x if x . ends_with ( " hours " ) = > FromStr ::from_str ( & x [ 0 .. x . len ( ) - 5 ] ) . unwrap_or_else ( bad ) * 60 * 60 ,
x if x . ends_with ( " days " ) = > FromStr ::from_str ( & x [ 0 .. x . len ( ) - 4 ] ) . unwrap_or_else ( bad ) * 24 * 60 * 60 ,
x = > FromStr ::from_str ( x ) . unwrap_or_else ( bad ) ,
} )
2016-07-10 13:18:33 +02:00
}
2016-07-08 17:26:06 +02:00
pub fn gas_pricer ( & self ) -> GasPricer {
2016-04-21 16:45:04 +02:00
match self . args . flag_gasprice . as_ref ( ) {
Some ( d ) = > {
2016-07-08 17:26:06 +02:00
GasPricer ::Fixed ( U256 ::from_dec_str ( d ) . unwrap_or_else ( | _ | {
2016-04-21 16:45:04 +02:00
die! ( " {}: Invalid gas price given. Must be a decimal unsigned 256-bit number. " , d )
2016-07-08 17:26:06 +02:00
} ) )
2016-04-21 16:45:04 +02:00
}
_ = > {
let usd_per_tx : f32 = FromStr ::from_str ( & self . args . flag_usd_per_tx ) . unwrap_or_else ( | _ | {
die! ( " {}: Invalid basic transaction price given in USD. Must be a decimal number. " , self . args . flag_usd_per_tx )
} ) ;
2016-07-08 17:26:06 +02:00
match self . args . flag_usd_per_eth . as_str ( ) {
" auto " = > {
GasPricer ::new_calibrated ( GasPriceCalibratorOptions {
usd_per_tx : usd_per_tx ,
recalibration_period : Self ::to_duration ( self . args . flag_price_update_period . as_str ( ) ) ,
} )
} ,
x = > {
let usd_per_eth : f32 = FromStr ::from_str ( x ) . unwrap_or_else ( | _ | die! ( " {}: Invalid ether price given in USD. Must be a decimal number. " , x ) ) ;
let wei_per_usd : f32 = 1.0e18 / usd_per_eth ;
let gas_per_tx : f32 = 21000.0 ;
let wei_per_gas : f32 = wei_per_usd * usd_per_tx / gas_per_tx ;
2016-07-15 10:11:14 +02:00
info! ( " Using a fixed conversion rate of Ξ1 = {} ({} wei/gas) " , White . bold ( ) . paint ( format! ( " US$ {} " , usd_per_eth ) ) , Yellow . bold ( ) . paint ( format! ( " {} " , wei_per_gas ) ) ) ;
2016-07-08 17:26:06 +02:00
GasPricer ::Fixed ( U256 ::from_dec_str ( & format! ( " {:.0} " , wei_per_gas ) ) . unwrap ( ) )
}
}
2016-04-21 16:45:04 +02:00
}
}
}
pub fn extra_data ( & self ) -> Bytes {
2016-06-25 14:35:43 +02:00
match self . args . flag_extradata . as_ref ( ) . or ( self . args . flag_extra_data . as_ref ( ) ) {
Some ( ref x ) if x . len ( ) < = 32 = > x . as_bytes ( ) . to_owned ( ) ,
None = > version_data ( ) ,
Some ( ref x ) = > { die! ( " {}: Extra data must be at most 32 characters. " , x ) ; }
2016-04-21 16:45:04 +02:00
}
}
pub fn spec ( & self ) -> Spec {
2016-04-30 19:58:28 +02:00
match self . chain ( ) . as_str ( ) {
2016-07-05 18:18:35 +02:00
" frontier " | " homestead " | " mainnet " = > ethereum ::new_frontier ( ) ,
2016-07-16 13:02:56 +02:00
" homestead-dogmatic " = > ethereum ::new_frontier_dogmatic ( ) ,
2016-04-21 16:45:04 +02:00
" morden " | " testnet " = > ethereum ::new_morden ( ) ,
" olympic " = > ethereum ::new_olympic ( ) ,
f = > Spec ::load ( contents ( f ) . unwrap_or_else ( | _ | {
die! ( " {}: Couldn't read chain specification file. Sure it exists? " , f )
} ) . as_ref ( ) ) ,
}
}
pub fn normalize_enode ( e : & str ) -> Option < String > {
if is_valid_node_url ( e ) {
Some ( e . to_owned ( ) )
} else {
None
}
}
pub fn init_nodes ( & self , spec : & Spec ) -> Vec < String > {
match self . args . flag_bootnodes {
Some ( ref x ) if ! x . is_empty ( ) = > x . split ( ',' ) . map ( | s | {
Self ::normalize_enode ( s ) . unwrap_or_else ( | | {
die! ( " {}: Invalid node address format given for a boot node. " , s )
} )
} ) . collect ( ) ,
Some ( _ ) = > Vec ::new ( ) ,
2016-07-12 10:28:35 +02:00
None = > spec . nodes ( ) . to_owned ( ) ,
2016-04-21 16:45:04 +02:00
}
}
2016-06-20 14:13:33 +02:00
pub fn init_reserved_nodes ( & self ) -> Vec < String > {
use std ::fs ::File ;
if let Some ( ref path ) = self . args . flag_reserved_peers {
let mut buffer = String ::new ( ) ;
let mut node_file = File ::open ( path ) . unwrap_or_else ( | e | {
die! ( " Error opening reserved nodes file: {} " , e ) ;
} ) ;
node_file . read_to_string ( & mut buffer ) . expect ( " Error reading reserved node file " ) ;
buffer . lines ( ) . map ( | s | {
Self ::normalize_enode ( s ) . unwrap_or_else ( | | {
die! ( " {}: Invalid node address format given for a reserved node. " , s ) ;
} )
} ) . collect ( )
} else {
Vec ::new ( )
}
}
2016-04-21 16:45:04 +02:00
pub fn net_addresses ( & self ) -> ( Option < SocketAddr > , Option < SocketAddr > ) {
2016-04-30 19:58:28 +02:00
let port = self . net_port ( ) ;
let listen_address = Some ( SocketAddr ::new ( IpAddr ::from_str ( " 0.0.0.0 " ) . unwrap ( ) , port ) ) ;
2016-04-21 16:45:04 +02:00
let public_address = if self . args . flag_nat . starts_with ( " extip: " ) {
let host = & self . args . flag_nat [ 6 .. ] ;
let host = IpAddr ::from_str ( host ) . unwrap_or_else ( | _ | die! ( " Invalid host given with `--nat extip:{}` " , host ) ) ;
2016-04-30 19:58:28 +02:00
Some ( SocketAddr ::new ( host , port ) )
2016-04-21 16:45:04 +02:00
} else {
2016-07-04 18:21:22 +02:00
None
2016-04-21 16:45:04 +02:00
} ;
( listen_address , public_address )
}
pub fn net_settings ( & self , spec : & Spec ) -> NetworkConfiguration {
let mut ret = NetworkConfiguration ::new ( ) ;
ret . nat_enabled = self . args . flag_nat = = " any " | | self . args . flag_nat = = " upnp " ;
ret . boot_nodes = self . init_nodes ( spec ) ;
let ( listen , public ) = self . net_addresses ( ) ;
ret . listen_address = listen ;
ret . public_address = public ;
2016-05-17 10:32:05 +02:00
ret . use_secret = self . args . flag_node_key . as_ref ( ) . map ( | s | Secret ::from_str ( s ) . unwrap_or_else ( | _ | s . sha3 ( ) ) ) ;
2016-04-21 16:45:04 +02:00
ret . discovery_enabled = ! self . args . flag_no_discovery & & ! self . args . flag_nodiscover ;
2016-04-30 19:58:28 +02:00
ret . ideal_peers = self . max_peers ( ) ;
2016-04-21 16:45:04 +02:00
let mut net_path = PathBuf ::from ( & self . path ( ) ) ;
net_path . push ( " network " ) ;
ret . config_path = Some ( net_path . to_str ( ) . unwrap ( ) . to_owned ( ) ) ;
2016-06-20 14:13:33 +02:00
ret . reserved_nodes = self . init_reserved_nodes ( ) ;
2016-06-21 13:56:33 +02:00
if self . args . flag_reserved_only {
ret . non_reserved_mode = ::util ::network ::NonReservedPeerMode ::Deny ;
}
2016-04-21 16:45:04 +02:00
ret
}
2016-07-11 09:46:33 +02:00
fn find_best_db ( & self , spec : & Spec ) -> Option < journaldb ::Algorithm > {
2016-04-21 16:45:04 +02:00
let mut ret = None ;
let mut latest_era = None ;
let jdb_types = [ journaldb ::Algorithm ::Archive , journaldb ::Algorithm ::EarlyMerge , journaldb ::Algorithm ::OverlayRecent , journaldb ::Algorithm ::RefCounted ] ;
for i in jdb_types . into_iter ( ) {
2016-06-27 14:25:50 +02:00
let db = journaldb ::new ( & append_path ( & get_db_path ( Path ::new ( & self . path ( ) ) , * i , spec . genesis_header ( ) . hash ( ) ) , " state " ) , * i , kvdb ::DatabaseConfig ::default ( ) ) ;
2016-04-21 16:45:04 +02:00
trace! ( target : " parity " , " Looking for best DB: {} at {:?} " , i , db . latest_era ( ) ) ;
match ( latest_era , db . latest_era ( ) ) {
( Some ( best ) , Some ( this ) ) if best > = this = > { }
( _ , None ) = > { }
( _ , Some ( this ) ) = > {
latest_era = Some ( this ) ;
ret = Some ( * i ) ;
}
}
}
ret
}
2016-07-11 09:46:33 +02:00
pub fn pruning_algorithm ( & self , spec : & Spec ) -> journaldb ::Algorithm {
match self . args . flag_pruning . as_str ( ) {
" archive " = > journaldb ::Algorithm ::Archive ,
" light " = > journaldb ::Algorithm ::EarlyMerge ,
" fast " = > journaldb ::Algorithm ::OverlayRecent ,
" basic " = > journaldb ::Algorithm ::RefCounted ,
" auto " = > self . find_best_db ( spec ) . unwrap_or ( journaldb ::Algorithm ::OverlayRecent ) ,
_ = > { die! ( " Invalid pruning method given. " ) ; }
}
}
2016-04-21 16:45:04 +02:00
pub fn client_config ( & self , spec : & Spec ) -> ClientConfig {
let mut client_config = ClientConfig ::default ( ) ;
2016-05-19 00:44:49 +02:00
2016-07-05 17:50:46 +02:00
client_config . mode = self . mode ( ) ;
2016-04-21 16:45:04 +02:00
match self . args . flag_cache {
Some ( mb ) = > {
client_config . blockchain . max_cache_size = mb * 1024 * 1024 ;
client_config . blockchain . pref_cache_size = client_config . blockchain . max_cache_size * 3 / 4 ;
}
None = > {
client_config . blockchain . pref_cache_size = self . args . flag_cache_pref_size ;
client_config . blockchain . max_cache_size = self . args . flag_cache_max_size ;
}
}
2016-06-20 12:42:04 +02:00
// forced blockchain (blocks + extras) db cache size if provided
client_config . blockchain . db_cache_size = self . args . flag_db_cache_size . and_then ( | cs | Some ( cs / 2 ) ) ;
2016-05-19 00:44:49 +02:00
2016-05-02 12:17:30 +02:00
client_config . tracing . enabled = match self . args . flag_tracing . as_str ( ) {
" auto " = > Switch ::Auto ,
" on " = > Switch ::On ,
" off " = > Switch ::Off ,
_ = > { die! ( " Invalid tracing method given! " ) }
} ;
2016-06-20 12:42:04 +02:00
// forced trace db cache size if provided
client_config . tracing . db_cache_size = self . args . flag_db_cache_size . and_then ( | cs | Some ( cs / 4 ) ) ;
2016-05-19 00:44:49 +02:00
2016-07-11 09:46:33 +02:00
client_config . pruning = self . pruning_algorithm ( spec ) ;
2016-05-19 00:44:49 +02:00
2016-07-01 20:29:56 +02:00
if self . args . flag_fat_db {
if let journaldb ::Algorithm ::Archive = client_config . pruning {
client_config . trie_spec = TrieSpec ::Fat ;
} else {
2016-07-06 18:28:11 +02:00
die! ( " Fatdb is not supported. Please re-run with --pruning=archive " )
2016-07-01 20:29:56 +02:00
}
}
2016-06-20 12:42:04 +02:00
// forced state db cache size if provided
client_config . db_cache_size = self . args . flag_db_cache_size . and_then ( | cs | Some ( cs / 4 ) ) ;
2016-06-27 13:58:12 +02:00
// compaction profile
2016-06-27 14:25:50 +02:00
client_config . db_compaction = match self . args . flag_db_compaction . as_str ( ) {
2016-06-27 19:22:54 +02:00
" ssd " = > DatabaseCompactionProfile ::Default ,
2016-06-27 13:58:12 +02:00
" hdd " = > DatabaseCompactionProfile ::HDD ,
2016-07-02 07:58:32 +02:00
_ = > { die! ( " Invalid compaction profile given (--db-compaction argument), expected hdd/ssd (default). " ) ; }
2016-06-27 13:58:12 +02:00
} ;
2016-05-19 00:44:49 +02:00
if self . args . flag_jitvm {
2016-07-06 18:28:11 +02:00
client_config . vm_type = VMType ::jit ( ) . unwrap_or_else ( | | die! ( " Parity is built without the JIT EVM. " ) )
2016-05-19 00:44:49 +02:00
}
2016-04-21 16:45:04 +02:00
trace! ( target : " parity " , " Using pruning strategy of {} " , client_config . pruning ) ;
client_config . name = self . args . flag_identity . clone ( ) ;
client_config . queue . max_mem_use = self . args . flag_queue_max_size ;
client_config
}
pub fn sync_config ( & self , spec : & Spec ) -> SyncConfig {
let mut sync_config = SyncConfig ::default ( ) ;
sync_config . network_id = self . args . flag_network_id . as_ref ( ) . or ( self . args . flag_networkid . as_ref ( ) ) . map_or ( spec . network_id ( ) , | id | {
U256 ::from_str ( id ) . unwrap_or_else ( | _ | die! ( " {}: Invalid index given with --network-id/--networkid " , id ) )
} ) ;
sync_config
}
2016-06-20 00:10:34 +02:00
pub fn account_service ( & self ) -> AccountProvider {
use ethcore ::ethstore ::{ import_accounts , EthStore } ;
use ethcore ::ethstore ::dir ::{ GethDirectory , DirectoryType , DiskDirectory } ;
2016-04-21 16:45:04 +02:00
// Secret Store
let passwords = self . args . flag_password . iter ( ) . flat_map ( | filename | {
BufReader ::new ( & File ::open ( filename ) . unwrap_or_else ( | _ | die! ( " {} Unable to read password file. Ensure it exists and permissions are correct. " , filename ) ) )
. lines ( )
. map ( | l | l . unwrap ( ) )
. collect ::< Vec < _ > > ( )
. into_iter ( )
} ) . collect ::< Vec < _ > > ( ) ;
2016-06-20 00:10:34 +02:00
if ! self . args . flag_no_import_keys {
2016-06-20 15:20:55 +02:00
let dir_type = if self . args . flag_testnet {
DirectoryType ::Testnet
} else {
DirectoryType ::Main
2016-06-20 00:10:34 +02:00
} ;
let from = GethDirectory ::open ( dir_type ) ;
let to = DiskDirectory ::create ( self . keys_path ( ) ) . unwrap ( ) ;
2016-06-20 16:29:04 +02:00
// ignore error, cause geth may not exist
let _ = import_accounts ( & from , & to ) ;
2016-06-20 00:10:34 +02:00
}
let dir = Box ::new ( DiskDirectory ::create ( self . keys_path ( ) ) . unwrap ( ) ) ;
let iterations = self . keys_iterations ( ) ;
let account_service = AccountProvider ::new ( Box ::new ( EthStore ::open_with_iterations ( dir , iterations ) . unwrap ( ) ) ) ;
2016-04-21 16:45:04 +02:00
if let Some ( ref unlocks ) = self . args . flag_unlock {
for d in unlocks . split ( ',' ) {
2016-05-17 10:32:05 +02:00
let a = Address ::from_str ( clean_0x ( d ) ) . unwrap_or_else ( | _ | {
2016-04-21 16:45:04 +02:00
die! ( " {}: Invalid address for --unlock. Must be 40 hex characters, without the 0x at the beginning. " , d )
} ) ;
2016-06-20 00:10:34 +02:00
if passwords . iter ( ) . find ( | p | account_service . unlock_account_permanently ( a , ( * p ) . clone ( ) ) . is_ok ( ) ) . is_none ( ) {
2016-04-21 16:45:04 +02:00
die! ( " No password given to unlock account {}. Pass the password using `--password`. " , a ) ;
}
}
}
account_service
}
2016-04-21 19:19:42 +02:00
2016-04-30 19:58:28 +02:00
pub fn rpc_apis ( & self ) -> String {
self . args . flag_rpcapi . clone ( ) . unwrap_or ( self . args . flag_jsonrpc_apis . clone ( ) )
}
2016-07-20 12:34:17 +02:00
pub fn rpc_cors ( & self ) -> Option < Vec < String > > {
2016-05-14 13:29:26 +02:00
let cors = self . args . flag_jsonrpc_cors . clone ( ) . or ( self . args . flag_rpccorsdomain . clone ( ) ) ;
2016-07-20 12:34:17 +02:00
cors . map ( | c | c . split ( ',' ) . map ( | s | s . to_owned ( ) ) . collect ( ) )
}
pub fn rpc_hosts ( & self ) -> Option < Vec < String > > {
match self . args . flag_jsonrpc_hosts . as_ref ( ) {
" none " = > return Some ( Vec ::new ( ) ) ,
" all " = > return None ,
_ = > { }
}
let hosts = self . args . flag_jsonrpc_hosts . split ( ',' ) . map ( | h | h . into ( ) ) . collect ( ) ;
Some ( hosts )
2016-04-30 19:58:28 +02:00
}
2016-05-13 12:53:33 +02:00
2016-06-07 17:14:03 +02:00
fn geth_ipc_path ( & self ) -> String {
2016-06-13 18:55:24 +02:00
if cfg! ( windows ) {
r "\\.\pipe\geth.ipc" . to_owned ( )
2016-07-07 13:18:09 +02:00
} else {
match self . args . flag_testnet {
true = > path ::ethereum ::with_testnet ( " geth.ipc " ) ,
false = > path ::ethereum ::with_default ( " geth.ipc " ) ,
} . to_str ( ) . unwrap ( ) . to_owned ( )
2016-06-13 18:55:24 +02:00
}
2016-05-13 12:53:33 +02:00
}
2016-05-14 19:43:29 +02:00
pub fn keys_iterations ( & self ) -> u32 {
self . args . flag_keys_iterations
}
2016-05-04 15:37:09 +02:00
pub fn ipc_settings ( & self ) -> IpcConfiguration {
IpcConfiguration {
2016-06-23 18:57:42 +02:00
enabled : ! ( self . args . flag_ipcdisable | | self . args . flag_ipc_off | | self . args . flag_no_ipc ) ,
2016-05-13 12:53:33 +02:00
socket_addr : self . ipc_path ( ) ,
2016-05-04 19:26:47 +02:00
apis : self . args . flag_ipcapi . clone ( ) . unwrap_or ( self . args . flag_ipc_apis . clone ( ) ) ,
2016-05-04 15:37:09 +02:00
}
}
2016-04-21 19:19:42 +02:00
pub fn network_settings ( & self ) -> NetworkSettings {
2016-05-04 19:26:47 +02:00
if self . args . flag_jsonrpc { println! ( " WARNING: Flag -j/--json-rpc is deprecated. JSON-RPC is now on by default. Ignoring. " ) ; }
2016-04-21 19:19:42 +02:00
NetworkSettings {
name : self . args . flag_identity . clone ( ) ,
2016-04-30 19:58:28 +02:00
chain : self . chain ( ) ,
max_peers : self . max_peers ( ) ,
network_port : self . net_port ( ) ,
2016-06-23 18:57:42 +02:00
rpc_enabled : ! self . args . flag_jsonrpc_off & & ! self . args . flag_no_jsonrpc ,
2016-04-21 19:19:42 +02:00
rpc_interface : self . args . flag_rpcaddr . clone ( ) . unwrap_or ( self . args . flag_jsonrpc_interface . clone ( ) ) ,
rpc_port : self . args . flag_rpcport . unwrap_or ( self . args . flag_jsonrpc_port ) ,
}
}
2016-05-13 17:32:32 +02:00
pub fn directories ( & self ) -> Directories {
let db_path = Configuration ::replace_home (
2016-05-17 10:32:05 +02:00
self . args . flag_datadir . as_ref ( ) . unwrap_or ( & self . args . flag_db_path ) ) ;
2016-05-13 17:36:18 +02:00
::std ::fs ::create_dir_all ( & db_path ) . unwrap_or_else ( | e | die_with_io_error ( " main " , e ) ) ;
2016-05-13 17:32:32 +02:00
2016-05-25 17:03:58 +02:00
let keys_path = Configuration ::replace_home (
if self . args . flag_testnet {
" $HOME/.parity/testnet_keys "
} else {
& self . args . flag_keys_path
}
) ;
2016-06-03 11:51:11 +02:00
::std ::fs ::create_dir_all ( & keys_path ) . unwrap_or_else ( | e | die_with_io_error ( " main " , e ) ) ;
let dapps_path = Configuration ::replace_home ( & self . args . flag_dapps_path ) ;
::std ::fs ::create_dir_all ( & dapps_path ) . unwrap_or_else ( | e | die_with_io_error ( " main " , e ) ) ;
2016-06-07 17:21:19 +02:00
let signer_path = Configuration ::replace_home ( & self . args . flag_signer_path ) ;
::std ::fs ::create_dir_all ( & signer_path ) . unwrap_or_else ( | e | die_with_io_error ( " main " , e ) ) ;
2016-07-04 11:53:21 +02:00
if self . args . flag_geth {
let geth_path = path ::ethereum ::default ( ) ;
::std ::fs ::create_dir_all ( geth_path . as_path ( ) ) . unwrap_or_else (
| e | die! ( " Error while attempting to create '{}' for geth mode: {} " , & geth_path . to_str ( ) . unwrap ( ) , e ) ) ;
}
2016-05-13 17:32:32 +02:00
Directories {
keys : keys_path ,
db : db_path ,
2016-06-03 11:51:11 +02:00
dapps : dapps_path ,
2016-06-07 17:21:19 +02:00
signer : signer_path ,
2016-05-13 17:32:32 +02:00
}
}
pub fn keys_path ( & self ) -> String {
self . directories ( ) . keys
}
pub fn path ( & self ) -> String {
self . directories ( ) . db
}
fn replace_home ( arg : & str ) -> String {
arg . replace ( " $HOME " , env ::home_dir ( ) . unwrap ( ) . to_str ( ) . unwrap ( ) )
}
fn ipc_path ( & self ) -> String {
2016-06-15 00:57:49 +02:00
if self . args . flag_geth {
self . geth_ipc_path ( )
} else if cfg! ( windows ) {
r "\\.\pipe\parity.jsonrpc" . to_owned ( )
} else {
Configuration ::replace_home ( & self . args . flag_ipcpath . clone ( ) . unwrap_or ( self . args . flag_ipc_path . clone ( ) ) )
2016-06-13 18:55:24 +02:00
}
}
pub fn have_color ( & self ) -> bool {
! self . args . flag_no_color & & ! cfg! ( windows )
2016-05-13 17:32:32 +02:00
}
2016-06-10 15:39:34 +02:00
pub fn signer_port ( & self ) -> Option < u16 > {
2016-06-24 14:29:15 +02:00
if ! self . signer_enabled ( ) {
2016-06-10 15:39:34 +02:00
None
2016-06-22 21:32:17 +02:00
} else {
Some ( self . args . flag_signer_port )
2016-06-10 15:39:34 +02:00
}
}
2016-06-24 12:10:36 +02:00
pub fn rpc_interface ( & self ) -> String {
match self . network_settings ( ) . rpc_interface . as_str ( ) {
" all " = > " 0.0.0.0 " ,
" local " = > " 127.0.0.1 " ,
x = > x ,
} . into ( )
}
pub fn dapps_interface ( & self ) -> String {
match self . args . flag_dapps_interface . as_str ( ) {
2016-07-19 13:15:25 +02:00
" local " = > " 127.0.0.1 " ,
2016-06-24 12:10:36 +02:00
x = > x ,
} . into ( )
}
2016-06-24 12:14:46 +02:00
pub fn dapps_enabled ( & self ) -> bool {
2016-07-13 11:10:43 +02:00
! self . args . flag_dapps_off & & ! self . args . flag_no_dapps & & cfg! ( feature = " dapps " )
2016-06-24 12:14:46 +02:00
}
2016-06-24 14:20:39 +02:00
pub fn signer_enabled ( & self ) -> bool {
2016-07-03 18:11:31 +02:00
( self . args . flag_unlock . is_none ( ) & & ! self . args . flag_no_signer ) | |
self . args . flag_force_signer
2016-06-24 14:20:39 +02:00
}
2016-04-21 16:45:04 +02:00
}
2016-04-30 19:58:28 +02:00
#[ cfg(test) ]
mod tests {
use super ::* ;
use cli ::USAGE ;
use docopt ::Docopt ;
use util ::network_settings ::NetworkSettings ;
fn parse ( args : & [ & str ] ) -> Configuration {
Configuration {
args : Docopt ::new ( USAGE ) . unwrap ( ) . argv ( args ) . decode ( ) . unwrap ( ) ,
}
}
#[ test ]
fn should_parse_network_settings ( ) {
// given
// when
let conf = parse ( & [ " parity " , " --testnet " , " --identity " , " testname " ] ) ;
// then
assert_eq! ( conf . network_settings ( ) , NetworkSettings {
name : " testname " . to_owned ( ) ,
chain : " morden " . to_owned ( ) ,
max_peers : 25 ,
network_port : 30303 ,
2016-05-04 22:09:30 +02:00
rpc_enabled : true ,
2016-04-30 19:58:28 +02:00
rpc_interface : " local " . to_owned ( ) ,
rpc_port : 8545 ,
} ) ;
}
#[ test ]
fn should_parse_rpc_settings_with_geth_compatiblity ( ) {
// given
fn assert ( conf : Configuration ) {
let net = conf . network_settings ( ) ;
assert_eq! ( net . rpc_enabled , true ) ;
assert_eq! ( net . rpc_interface , " all " . to_owned ( ) ) ;
assert_eq! ( net . rpc_port , 8000 ) ;
2016-07-20 12:34:17 +02:00
assert_eq! ( conf . rpc_cors ( ) , Some ( vec! [ " * " . to_owned ( ) ] ) ) ;
2016-04-30 19:58:28 +02:00
assert_eq! ( conf . rpc_apis ( ) , " web3,eth " . to_owned ( ) ) ;
}
// when
let conf1 = parse ( & [ " parity " , " -j " ,
" --jsonrpc-port " , " 8000 " ,
" --jsonrpc-interface " , " all " ,
" --jsonrpc-cors " , " * " ,
" --jsonrpc-apis " , " web3,eth "
] ) ;
let conf2 = parse ( & [ " parity " , " --rpc " ,
" --rpcport " , " 8000 " ,
" --rpcaddr " , " all " ,
" --rpccorsdomain " , " * " ,
" --rpcapi " , " web3,eth "
] ) ;
// then
assert ( conf1 ) ;
assert ( conf2 ) ;
}
2016-07-20 12:34:17 +02:00
#[ test ]
fn should_parse_rpc_hosts ( ) {
// given
// when
let conf0 = parse ( & [ " parity " ] ) ;
let conf1 = parse ( & [ " parity " , " --jsonrpc-hosts " , " none " ] ) ;
let conf2 = parse ( & [ " parity " , " --jsonrpc-hosts " , " all " ] ) ;
let conf3 = parse ( & [ " parity " , " --jsonrpc-hosts " , " ethcore.io,something.io " ] ) ;
// then
assert_eq! ( conf0 . rpc_hosts ( ) , Some ( Vec ::new ( ) ) ) ;
assert_eq! ( conf1 . rpc_hosts ( ) , Some ( Vec ::new ( ) ) ) ;
assert_eq! ( conf2 . rpc_hosts ( ) , None ) ;
assert_eq! ( conf3 . rpc_hosts ( ) , Some ( vec! [ " ethcore.io " . into ( ) , " something.io " . into ( ) ] ) ) ;
}
2016-04-30 19:58:28 +02:00
}