// 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 . use std::collections::BTreeMap; use std::str::FromStr; use std::sync::Arc; use std::net::SocketAddr; use ethcore::client::Client; use ethsync::EthSync; use ethminer::{Miner, ExternalMiner}; use util::RotatingLogger; use util::panics::PanicHandler; use util::keys::store::AccountService; use util::network_settings::NetworkSettings; use die::*; use jsonipc; #[cfg(feature = "rpc")] pub use ethcore_rpc::Server as RpcServer; #[cfg(feature = "rpc")] use ethcore_rpc::{RpcServerError, RpcServer as Server}; #[cfg(not(feature = "rpc"))] pub struct RpcServer; pub struct HttpConfiguration { pub enabled: bool, pub interface: String, pub port: u16, pub apis: String, pub cors: Vec, } pub struct IpcConfiguration { pub enabled: bool, pub socket_addr: String, pub apis: String, } pub struct Dependencies { pub panic_handler: Arc, pub client: Arc, pub sync: Arc, pub secret_store: Arc, pub miner: Arc, pub external_miner: Arc, pub logger: Arc, pub settings: Arc, } pub fn new_http(conf: HttpConfiguration, deps: &Arc) -> Option { if !conf.enabled { return None; } let interface = match conf.interface.as_str() { "all" => "0.0.0.0", "local" => "127.0.0.1", x => x, }; let apis = conf.apis.split(',').collect(); let url = format!("{}:{}", interface, conf.port); let addr = SocketAddr::from_str(&url).unwrap_or_else(|_| die!("{}: Invalid JSONRPC listen host/port given.", url)); Some(setup_http_rpc_server(deps, &addr, conf.cors, apis)) } pub fn new_ipc(conf: IpcConfiguration, deps: &Arc) -> Option { if !conf.enabled { return None; } let apis = conf.apis.split(',').collect(); Some(setup_ipc_rpc_server(deps, &conf.socket_addr, apis)) } fn setup_rpc_server(apis: Vec<&str>, deps: &Arc) -> Server { use ethcore_rpc::v1::*; let server = Server::new(); let mut modules = BTreeMap::new(); for api in apis.into_iter() { match api { "web3" => { modules.insert("web3".to_owned(), "1.0".to_owned()); server.add_delegate(Web3Client::new().to_delegate()); }, "net" => { modules.insert("net".to_owned(), "1.0".to_owned()); server.add_delegate(NetClient::new(&deps.sync).to_delegate()); }, "eth" => { modules.insert("eth".to_owned(), "1.0".to_owned()); server.add_delegate(EthClient::new(&deps.client, &deps.sync, &deps.secret_store, &deps.miner, &deps.external_miner).to_delegate()); server.add_delegate(EthFilterClient::new(&deps.client, &deps.miner).to_delegate()); }, "personal" => { modules.insert("personal".to_owned(), "1.0".to_owned()); server.add_delegate(PersonalClient::new(&deps.secret_store, &deps.client, &deps.miner).to_delegate()) }, "ethcore" => { modules.insert("ethcore".to_owned(), "1.0".to_owned()); server.add_delegate(EthcoreClient::new(&deps.client, &deps.miner, deps.logger.clone(), deps.settings.clone()).to_delegate()) }, "traces" => { modules.insert("traces".to_owned(), "1.0".to_owned()); server.add_delegate(TracesClient::new(&deps.client).to_delegate()) }, _ => { die!("{}: Invalid API name to be enabled.", api); }, } } server.add_delegate(RpcClient::new(modules).to_delegate()); server } #[cfg(not(feature = "rpc"))] pub fn setup_http_rpc_server( _deps: Dependencies, _url: &SocketAddr, _cors_domain: Option, _apis: Vec<&str>, ) -> ! { die!("Your Parity version has been compiled without JSON-RPC support.") } #[cfg(feature = "rpc")] pub fn setup_http_rpc_server( dependencies: &Arc, url: &SocketAddr, cors_domains: Vec, apis: Vec<&str>, ) -> RpcServer { let server = setup_rpc_server(apis, dependencies); let start_result = server.start_http(url, cors_domains); let deps = dependencies.clone(); match start_result { Err(RpcServerError::IoError(err)) => die_with_io_error("RPC", err), Err(e) => die!("RPC: {:?}", e), Ok(server) => { server.set_panic_handler(move || { deps.panic_handler.notify_all("Panic in RPC thread.".to_owned()); }); server }, } } pub fn setup_ipc_rpc_server(dependencies: &Arc, addr: &str, apis: Vec<&str>) -> jsonipc::Server { let server = setup_rpc_server(apis, dependencies); match server.start_ipc(addr) { Err(jsonipc::Error::Io(io_error)) => die_with_io_error("RPC", io_error), Err(any_error) => die!("RPC: {:?}", any_error), Ok(server) => server } }