2019-01-07 11:33:07 +01:00
// Copyright 2015-2019 Parity Technologies (UK) Ltd.
// This file is part of Parity Ethereum.
2016-04-21 13:12:43 +02:00
2019-01-07 11:33:07 +01:00
// Parity Ethereum is free software: you can redistribute it and/or modify
2016-04-21 13:12:43 +02: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,
2016-04-21 13:12:43 +02: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/>.
2016-04-21 13:12:43 +02:00
2020-08-05 06:08:03 +02:00
use std ::{ collections ::HashSet , io , path ::PathBuf , sync ::Arc } ;
2017-01-11 20:02:27 +01:00
2020-08-05 06:08:03 +02:00
use dir ::{ default_data_path , helpers ::replace_home } ;
2017-12-24 09:34:43 +01:00
use helpers ::parity_ipc_path ;
2017-05-23 12:26:39 +02:00
use jsonrpc_core ::MetaIoHandler ;
2020-08-05 06:08:03 +02:00
use parity_rpc ::{
self as rpc ,
informant ::{ Middleware , RpcStats } ,
DomainsValidation , Metadata ,
} ;
2018-10-22 09:40:50 +02:00
use parity_runtime ::Executor ;
2017-04-03 10:27:37 +02:00
use rpc_apis ::{ self , ApiSet } ;
2016-04-21 13:12:43 +02:00
2020-08-05 06:08:03 +02:00
pub use parity_rpc ::{ HttpServer , IpcServer , RequestMiddleware } ;
2019-11-11 21:57:38 +01:00
//pub use parity_rpc::ws::Server as WsServer;
2020-08-05 06:08:03 +02:00
pub use parity_rpc ::ws ::{ ws , Server as WsServer } ;
2016-04-21 13:12:43 +02:00
2017-05-24 12:24:07 +02:00
pub const DAPPS_DOMAIN : & 'static str = " web3.site " ;
2017-04-03 10:27:37 +02:00
#[ derive(Debug, Clone, PartialEq) ]
2016-05-04 15:37:09 +02:00
pub struct HttpConfiguration {
2020-08-05 06:08:03 +02:00
pub enabled : bool ,
pub interface : String ,
pub port : u16 ,
pub apis : ApiSet ,
pub cors : Option < Vec < String > > ,
pub hosts : Option < Vec < String > > ,
pub server_threads : usize ,
pub processing_threads : usize ,
pub max_payload : usize ,
pub keep_alive : bool ,
2016-04-21 13:57:27 +02:00
}
2016-07-25 16:09:47 +02:00
impl Default for HttpConfiguration {
2020-08-05 06:08:03 +02:00
fn default ( ) -> Self {
HttpConfiguration {
enabled : true ,
interface : " 127.0.0.1 " . into ( ) ,
port : 8545 ,
apis : ApiSet ::UnsafeContext ,
cors : Some ( vec! [ ] ) ,
hosts : Some ( vec! [ ] ) ,
server_threads : 1 ,
processing_threads : 4 ,
max_payload : 5 ,
keep_alive : true ,
}
}
2016-07-25 16:09:47 +02:00
}
#[ derive(Debug, PartialEq) ]
2016-05-04 15:37:09 +02:00
pub struct IpcConfiguration {
2020-08-05 06:08:03 +02:00
pub enabled : bool ,
pub socket_addr : String ,
pub apis : ApiSet ,
2016-07-25 16:09:47 +02:00
}
impl Default for IpcConfiguration {
2020-08-05 06:08:03 +02:00
fn default ( ) -> Self {
IpcConfiguration {
enabled : true ,
socket_addr : if cfg! ( windows ) {
r "\\.\pipe\jsonrpc.ipc" . into ( )
} else {
let data_dir = ::dir ::default_data_path ( ) ;
parity_ipc_path ( & data_dir , " $BASE/jsonrpc.ipc " , 0 )
} ,
apis : ApiSet ::IpcContext ,
}
}
2016-05-04 15:37:09 +02:00
}
2017-05-24 12:24:07 +02:00
#[ derive(Debug, Clone, PartialEq) ]
2017-04-13 16:32:07 +02:00
pub struct WsConfiguration {
2020-08-05 06:08:03 +02:00
pub enabled : bool ,
pub interface : String ,
pub port : u16 ,
pub apis : ApiSet ,
pub max_connections : usize ,
pub origins : Option < Vec < String > > ,
pub hosts : Option < Vec < String > > ,
pub signer_path : PathBuf ,
pub support_token_api : bool ,
2017-04-13 16:32:07 +02:00
}
impl Default for WsConfiguration {
2020-08-05 06:08:03 +02:00
fn default ( ) -> Self {
let data_dir = default_data_path ( ) ;
WsConfiguration {
enabled : true ,
interface : " 127.0.0.1 " . into ( ) ,
port : 8546 ,
apis : ApiSet ::UnsafeContext ,
max_connections : 100 ,
origins : Some ( vec! [
" parity://* " . into ( ) ,
" chrome-extension://* " . into ( ) ,
" moz-extension://* " . into ( ) ,
] ) ,
hosts : Some ( Vec ::new ( ) ) ,
signer_path : replace_home ( & data_dir , " $BASE/signer " ) . into ( ) ,
support_token_api : true ,
}
}
2016-06-13 18:55:24 +02:00
}
2017-05-24 12:24:07 +02:00
impl WsConfiguration {
2020-08-05 06:08:03 +02:00
pub fn address ( & self ) -> Option < rpc ::Host > {
address ( self . enabled , & self . interface , self . port , & self . hosts )
}
2017-09-21 14:52:44 +02:00
}
2020-08-05 06:08:03 +02:00
fn address (
enabled : bool ,
bind_iface : & str ,
bind_port : u16 ,
hosts : & Option < Vec < String > > ,
) -> Option < rpc ::Host > {
if ! enabled {
return None ;
}
match * hosts {
Some ( ref hosts ) if ! hosts . is_empty ( ) = > Some ( hosts [ 0 ] . clone ( ) . into ( ) ) ,
_ = > Some ( format! ( " {} : {} " , bind_iface , bind_port ) . into ( ) ) ,
}
2017-04-13 16:32:07 +02:00
}
2017-05-24 12:24:07 +02:00
pub struct Dependencies < D : rpc_apis ::Dependencies > {
2020-08-05 06:08:03 +02:00
pub apis : Arc < D > ,
pub executor : Executor ,
pub stats : Arc < RpcStats > ,
2017-04-03 10:27:37 +02:00
}
2017-04-13 16:32:07 +02:00
pub fn new_ws < D : rpc_apis ::Dependencies > (
2020-08-05 06:08:03 +02:00
conf : WsConfiguration ,
deps : & Dependencies < D > ,
2017-04-13 16:32:07 +02:00
) -> Result < Option < WsServer > , String > {
2020-08-05 06:08:03 +02:00
if ! conf . enabled {
return Ok ( None ) ;
}
let domain = DAPPS_DOMAIN ;
let url = format! ( " {} : {} " , conf . interface , conf . port ) ;
let addr = url
. parse ( )
. map_err ( | _ | format! ( " Invalid WebSockets listen host/port given: {} " , url ) ) ? ;
let full_handler = setup_apis ( rpc_apis ::ApiSet ::All , deps ) ;
let handler = {
let mut handler = MetaIoHandler ::with_middleware ( (
rpc ::WsDispatcher ::new ( full_handler ) ,
Middleware ::new ( deps . stats . clone ( ) , deps . apis . activity_notifier ( ) ) ,
) ) ;
let apis = conf . apis . list_apis ( ) ;
deps . apis . extend_with_set ( & mut handler , & apis ) ;
handler
} ;
let allowed_origins = into_domains ( with_domain ( conf . origins , domain , & None ) ) ;
let allowed_hosts = into_domains ( with_domain ( conf . hosts , domain , & Some ( url . clone ( ) . into ( ) ) ) ) ;
let signer_path ;
let path = match conf . support_token_api {
true = > {
signer_path = ::signer ::codes_path ( & conf . signer_path ) ;
Some ( signer_path . as_path ( ) )
}
false = > None ,
} ;
let start_result = rpc ::start_ws (
& addr ,
handler ,
allowed_origins ,
allowed_hosts ,
conf . max_connections ,
rpc ::WsExtractor ::new ( path . clone ( ) ) ,
rpc ::WsExtractor ::new ( path . clone ( ) ) ,
rpc ::WsStats ::new ( deps . stats . clone ( ) ) ,
) ;
// match start_result {
// Ok(server) => Ok(Some(server)),
// Err(rpc::ws::Error::Io(rpc::ws::ErrorKind::Io(ref err), _)) if err.kind() == io::ErrorKind::AddrInUse => Err(
// format!("WebSockets address {} is already in use, make sure that another instance of an Ethereum client is not running or change the address using the --ws-port and --ws-interface options.", url)
// ),
// Err(e) => Err(format!("WebSockets error: {:?}", e)),
// }
match start_result {
2017-04-13 16:32:07 +02:00
Ok ( server ) = > Ok ( Some ( server ) ) ,
2019-11-11 21:57:38 +01:00
Err ( rpc ::ws ::Error ::WsError ( ws ::Error {
kind : ws ::ErrorKind ::Io ( ref err ) , ..
} ) ) if err . kind ( ) = = io ::ErrorKind ::AddrInUse = > Err (
2017-04-13 16:32:07 +02:00
format! ( " WebSockets address {} is already in use, make sure that another instance of an Ethereum client is not running or change the address using the --ws-port and --ws-interface options. " , url )
) ,
Err ( e ) = > Err ( format! ( " WebSockets error: {:?} " , e ) ) ,
}
}
2017-04-03 12:44:52 +02:00
pub fn new_http < D : rpc_apis ::Dependencies > (
2020-08-05 06:08:03 +02:00
id : & str ,
options : & str ,
conf : HttpConfiguration ,
deps : & Dependencies < D > ,
2017-04-03 12:44:52 +02:00
) -> Result < Option < HttpServer > , String > {
2020-08-05 06:08:03 +02:00
if ! conf . enabled {
return Ok ( None ) ;
}
let domain = DAPPS_DOMAIN ;
let url = format! ( " {} : {} " , conf . interface , conf . port ) ;
let addr = url
. parse ( )
. map_err ( | _ | format! ( " Invalid {} listen host/port given: {} " , id , url ) ) ? ;
let handler = setup_apis ( conf . apis , deps ) ;
let cors_domains = into_domains ( conf . cors ) ;
let allowed_hosts = into_domains ( with_domain ( conf . hosts , domain , & Some ( url . clone ( ) . into ( ) ) ) ) ;
let start_result = rpc ::start_http (
& addr ,
cors_domains ,
allowed_hosts ,
handler ,
rpc ::RpcExtractor ,
conf . server_threads ,
conf . max_payload ,
conf . keep_alive ,
) ;
match start_result {
2017-04-03 10:27:37 +02:00
Ok ( server ) = > Ok ( Some ( server ) ) ,
2017-11-14 11:38:17 +01:00
Err ( ref err ) if err . kind ( ) = = io ::ErrorKind ::AddrInUse = > Err (
2017-05-24 12:24:07 +02:00
format! ( " {} address {} is already in use, make sure that another instance of an Ethereum client is not running or change the address using the -- {} -port and -- {} -interface options. " , id , url , options , options )
2017-04-13 16:32:07 +02:00
) ,
2017-05-24 12:24:07 +02:00
Err ( e ) = > Err ( format! ( " {} error: {:?} " , id , e ) ) ,
2016-04-21 13:12:43 +02:00
}
}
2016-06-13 18:55:24 +02:00
2017-04-03 12:44:52 +02:00
pub fn new_ipc < D : rpc_apis ::Dependencies > (
2020-08-05 06:08:03 +02:00
conf : IpcConfiguration ,
dependencies : & Dependencies < D > ,
2017-04-03 12:44:52 +02:00
) -> Result < Option < IpcServer > , String > {
2020-08-05 06:08:03 +02:00
if ! conf . enabled {
return Ok ( None ) ;
}
let handler = setup_apis ( conf . apis , dependencies ) ;
let path = PathBuf ::from ( & conf . socket_addr ) ;
// Make sure socket file can be created on unix-like OS.
// Windows pipe paths are not on the FS.
if ! cfg! ( windows ) {
if let Some ( dir ) = path . parent ( ) {
::std ::fs ::create_dir_all ( & dir ) . map_err ( | err | {
format! (
" Unable to create IPC directory at {}: {} " ,
dir . display ( ) ,
err
)
} ) ? ;
}
}
match rpc ::start_ipc ( & conf . socket_addr , handler , rpc ::RpcExtractor ) {
Ok ( server ) = > Ok ( Some ( server ) ) ,
Err ( io_error ) = > Err ( format! ( " IPC error: {} " , io_error ) ) ,
}
2017-04-03 10:27:37 +02:00
}
2017-05-24 12:24:07 +02:00
fn into_domains < T : From < String > > ( items : Option < Vec < String > > ) -> DomainsValidation < T > {
2020-08-05 06:08:03 +02:00
items
. map ( | vals | vals . into_iter ( ) . map ( T ::from ) . collect ( ) )
. into ( )
2017-05-24 12:24:07 +02:00
}
2017-04-03 10:27:37 +02:00
2020-08-05 06:08:03 +02:00
fn with_domain (
items : Option < Vec < String > > ,
domain : & str ,
dapps_address : & Option < rpc ::Host > ,
) -> Option < Vec < String > > {
fn extract_port ( s : & str ) -> Option < u16 > {
s . split ( ':' ) . nth ( 1 ) . and_then ( | s | s . parse ( ) . ok ( ) )
}
items . map ( move | items | {
let mut items = items . into_iter ( ) . collect ::< HashSet < _ > > ( ) ;
{
let mut add_hosts = | address : & Option < rpc ::Host > | {
if let Some ( host ) = address . clone ( ) {
items . insert ( host . to_string ( ) ) ;
items . insert ( host . replace ( " 127.0.0.1 " , " localhost " ) ) ;
items . insert ( format! ( " http://*. {} " , domain ) ) ; //proxypac
if let Some ( port ) = extract_port ( & * host ) {
items . insert ( format! ( " http://*. {} : {} " , domain , port ) ) ;
}
}
} ;
add_hosts ( dapps_address ) ;
}
items . into_iter ( ) . collect ( )
} )
2017-05-24 12:24:07 +02:00
}
2017-04-03 10:27:37 +02:00
2020-08-05 06:08:03 +02:00
pub fn setup_apis < D > (
apis : ApiSet ,
deps : & Dependencies < D > ,
) -> MetaIoHandler < Metadata , Middleware < D ::Notifier > >
where
D : rpc_apis ::Dependencies ,
2017-05-24 12:24:07 +02:00
{
2020-08-05 06:08:03 +02:00
let mut handler = MetaIoHandler ::with_middleware ( Middleware ::new (
deps . stats . clone ( ) ,
deps . apis . activity_notifier ( ) ,
) ) ;
let apis = apis . list_apis ( ) ;
deps . apis . extend_with_set ( & mut handler , & apis ) ;
handler
2016-05-04 15:37:09 +02:00
}
2017-09-21 14:52:44 +02:00
#[ cfg(test) ]
mod tests {
2020-08-05 06:08:03 +02:00
use super ::address ;
#[ test ]
fn should_return_proper_address ( ) {
assert_eq! ( address ( false , " localhost " , 8180 , & None ) , None ) ;
assert_eq! (
address ( true , " localhost " , 8180 , & None ) ,
Some ( " localhost:8180 " . into ( ) )
) ;
assert_eq! (
address ( true , " localhost " , 8180 , & Some ( vec! [ " host:443 " . into ( ) ] ) ) ,
Some ( " host:443 " . into ( ) )
) ;
assert_eq! (
address ( true , " localhost " , 8180 , & Some ( vec! [ " host " . into ( ) ] ) ) ,
Some ( " host " . into ( ) )
) ;
}
2017-09-21 14:52:44 +02:00
}