// 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 . //! Account management (personal) rpc implementation use std::sync::{Arc, Weak}; use std::collections::{BTreeMap}; use jsonrpc_core::*; use v1::traits::Personal; use v1::types::{H160 as RpcH160, TransactionRequest}; use v1::impls::unlock_sign_and_dispatch; use v1::helpers::{TransactionRequest as TRequest}; use ethcore::account_provider::AccountProvider; use util::Address; use ethcore::client::MiningBlockChainClient; use ethcore::miner::MinerService; /// Account management (personal) rpc implementation. pub struct PersonalClient where C: MiningBlockChainClient, M: MinerService { accounts: Weak, client: Weak, miner: Weak, signer_port: Option, } impl PersonalClient where C: MiningBlockChainClient, M: MinerService { /// Creates new PersonalClient pub fn new(store: &Arc, client: &Arc, miner: &Arc, signer_port: Option) -> Self { PersonalClient { accounts: Arc::downgrade(store), client: Arc::downgrade(client), miner: Arc::downgrade(miner), signer_port: signer_port, } } fn active(&self) -> Result<(), Error> { // TODO: only call every 30s at most. take_weak!(self.client).keep_alive(); Ok(()) } } impl Personal for PersonalClient where C: MiningBlockChainClient, M: MinerService { fn signer_enabled(&self, _: Params) -> Result { try!(self.active()); self.signer_port .map(|v| to_value(&v)) .unwrap_or_else(|| to_value(&false)) } fn accounts(&self, _: Params) -> Result { try!(self.active()); let store = take_weak!(self.accounts); to_value(&store.accounts().into_iter().map(Into::into).collect::>()) } fn new_account(&self, params: Params) -> Result { try!(self.active()); from_params::<(String, )>(params).and_then( |(pass, )| { let store = take_weak!(self.accounts); match store.new_account(&pass) { Ok(address) => to_value(&RpcH160::from(address)), Err(_) => Err(Error::internal_error()) } } ) } fn unlock_account(&self, params: Params) -> Result { try!(self.active()); from_params::<(RpcH160, String, u64)>(params).and_then( |(account, account_pass, _)|{ let account: Address = account.into(); let store = take_weak!(self.accounts); match store.unlock_account_temporarily(account, account_pass) { Ok(_) => Ok(Value::Bool(true)), Err(_) => Ok(Value::Bool(false)), } }) } fn sign_and_send_transaction(&self, params: Params) -> Result { try!(self.active()); from_params::<(TransactionRequest, String)>(params) .and_then(|(request, password)| { let request: TRequest = request.into(); let sender = request.from; let accounts = take_weak!(self.accounts); unlock_sign_and_dispatch(&*take_weak!(self.client), &*take_weak!(self.miner), request, &*accounts, sender, password) }) } fn set_account_name(&self, params: Params) -> Result { try!(self.active()); let store = take_weak!(self.accounts); from_params::<(RpcH160, _)>(params).and_then(|(addr, name)| { let addr: Address = addr.into(); store.set_account_name(addr, name).map_err(|_| Error::invalid_params()).map(|_| Value::Null) }) } fn set_account_meta(&self, params: Params) -> Result { try!(self.active()); let store = take_weak!(self.accounts); from_params::<(RpcH160, _)>(params).and_then(|(addr, meta)| { let addr: Address = addr.into(); store.set_account_meta(addr, meta).map_err(|_| Error::invalid_params()).map(|_| Value::Null) }) } fn accounts_info(&self, _: Params) -> Result { try!(self.active()); let store = take_weak!(self.accounts); Ok(Value::Object(try!(store.accounts_info().map_err(|_| Error::invalid_params())).into_iter().map(|(a, v)| { let m = map![ "name".to_owned() => to_value(&v.name).unwrap(), "meta".to_owned() => to_value(&v.meta).unwrap(), "uuid".to_owned() => if let &Some(ref uuid) = &v.uuid { to_value(uuid).unwrap() } else { Value::Null } ]; (format!("0x{}", a.hex()), Value::Object(m)) }).collect::>())) } }