// 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 . //! Traces api implementation. use std::sync::{Weak, Arc}; use jsonrpc_core::*; use std::collections::BTreeMap; use util::{H256, U256, FixedHash, Uint}; use serde; use ethcore::client::{BlockChainClient, CallAnalytics, TransactionID, TraceId}; use ethcore::trace::VMTrace; use ethcore::miner::MinerService; use ethcore::state_diff::StateDiff; use ethcore::account_diff::{Diff, Existance}; use ethcore::transaction::{Transaction as EthTransaction, SignedTransaction, Action}; use v1::traits::Traces; use v1::types::{TraceFilter, LocalizedTrace, Trace, BlockNumber, Index, CallRequest, Bytes}; /// Traces api implementation. pub struct TracesClient where C: BlockChainClient, M: MinerService { client: Weak, miner: Weak, } impl TracesClient where C: BlockChainClient, M: MinerService { /// Creates new Traces client. pub fn new(client: &Arc, miner: &Arc) -> Self { TracesClient { client: Arc::downgrade(client), miner: Arc::downgrade(miner), } } // TODO: share with eth.rs fn sign_call(&self, request: CallRequest) -> Result { let client = take_weak!(self.client); let miner = take_weak!(self.miner); let from = request.from.unwrap_or(0.into()); Ok(EthTransaction { nonce: request.nonce.unwrap_or_else(|| client.latest_nonce(&from)), action: request.to.map_or(Action::Create, Action::Call), gas: request.gas.unwrap_or(50_000_000.into()), gas_price: request.gas_price.unwrap_or_else(|| miner.sensible_gas_price()), value: request.value.unwrap_or(0.into()), data: request.data.map_or_else(Vec::new, |d| d.to_vec()) }.fake_sign(from)) } } fn vm_trace_to_object(t: &VMTrace) -> Value { let mut ret = BTreeMap::new(); ret.insert("code".to_owned(), to_value(&t.code).unwrap()); let mut subs = t.subs.iter(); let mut next_sub = subs.next(); let ops = t.operations .iter() .enumerate() .map(|(i, op)| { let mut m = map![ "pc".to_owned() => to_value(&op.pc).unwrap(), "cost".to_owned() => match op.gas_cost <= U256::from(!0u64) { true => to_value(&op.gas_cost.low_u64()), false => to_value(&op.gas_cost), }.unwrap() ]; if let Some(ref ex) = op.executed { let mut em = map![ "used".to_owned() => to_value(&ex.gas_used.low_u64()).unwrap(), "push".to_owned() => to_value(&ex.stack_push).unwrap() ]; if let Some(ref md) = ex.mem_diff { em.insert("mem".to_owned(), Value::Object(map![ "off".to_owned() => to_value(&md.offset).unwrap(), "data".to_owned() => to_value(&md.data).unwrap() ])); } if let Some(ref sd) = ex.store_diff { em.insert("store".to_owned(), Value::Object(map![ "key".to_owned() => to_value(&sd.location).unwrap(), "val".to_owned() => to_value(&sd.value).unwrap() ])); } m.insert("ex".to_owned(), Value::Object(em)); } if next_sub.is_some() && next_sub.unwrap().parent_step == i { m.insert("sub".to_owned(), vm_trace_to_object(next_sub.unwrap())); next_sub = subs.next(); } Value::Object(m) }) .collect::>(); ret.insert("ops".to_owned(), Value::Array(ops)); Value::Object(ret) } fn diff_to_object(d: &Diff) -> Value where T: serde::Serialize + Eq { let mut ret = BTreeMap::new(); match *d { Diff::Same => { ret.insert("diff".to_owned(), Value::String("=".to_owned())); } Diff::Born(ref x) => { ret.insert("diff".to_owned(), Value::String("+".to_owned())); ret.insert("+".to_owned(), to_value(x).unwrap()); } Diff::Died(ref x) => { ret.insert("diff".to_owned(), Value::String("-".to_owned())); ret.insert("-".to_owned(), to_value(x).unwrap()); } Diff::Changed(ref from, ref to) => { ret.insert("diff".to_owned(), Value::String("*".to_owned())); ret.insert("-".to_owned(), to_value(from).unwrap()); ret.insert("+".to_owned(), to_value(to).unwrap()); } }; Value::Object(ret) } fn state_diff_to_object(t: &StateDiff) -> Value { Value::Object(t.iter().map(|(address, account)| { (address.hex(), Value::Object(map![ "existance".to_owned() => Value::String(match account.existance() { Existance::Born => "+", Existance::Alive => ".", Existance::Died => "-", }.to_owned()), "balance".to_owned() => diff_to_object(&account.balance), "nonce".to_owned() => diff_to_object(&account.nonce), "code".to_owned() => diff_to_object(&account.code), "storage".to_owned() => Value::Object(account.storage.iter().map(|(key, val)| { (key.hex(), diff_to_object(&val)) }).collect::>()) ])) }).collect::>()) } impl Traces for TracesClient where C: BlockChainClient + 'static, M: MinerService + 'static { fn filter(&self, params: Params) -> Result { from_params::<(TraceFilter,)>(params) .and_then(|(filter, )| { let client = take_weak!(self.client); let traces = client.filter_traces(filter.into()); let traces = traces.map_or_else(Vec::new, |traces| traces.into_iter().map(LocalizedTrace::from).collect()); to_value(&traces) }) } fn block_traces(&self, params: Params) -> Result { from_params::<(BlockNumber,)>(params) .and_then(|(block_number,)| { let client = take_weak!(self.client); let traces = client.block_traces(block_number.into()); let traces = traces.map_or_else(Vec::new, |traces| traces.into_iter().map(LocalizedTrace::from).collect()); to_value(&traces) }) } fn transaction_traces(&self, params: Params) -> Result { from_params::<(H256,)>(params) .and_then(|(transaction_hash,)| { let client = take_weak!(self.client); let traces = client.transaction_traces(TransactionID::Hash(transaction_hash)); let traces = traces.map_or_else(Vec::new, |traces| traces.into_iter().map(LocalizedTrace::from).collect()); to_value(&traces) }) } fn trace(&self, params: Params) -> Result { from_params::<(H256, Vec)>(params) .and_then(|(transaction_hash, address)| { let client = take_weak!(self.client); let id = TraceId { transaction: TransactionID::Hash(transaction_hash), address: address.into_iter().map(|i| i.value()).collect() }; let trace = client.trace(id); let trace = trace.map(LocalizedTrace::from); to_value(&trace) }) } fn call(&self, params: Params) -> Result { trace!(target: "jsonrpc", "call: {:?}", params); from_params(params) .and_then(|(request, flags)| { let flags: Vec = flags; let analytics = CallAnalytics { transaction_tracing: flags.contains(&("trace".to_owned())), vm_tracing: flags.contains(&("vmTrace".to_owned())), state_diffing: flags.contains(&("stateDiff".to_owned())), }; let signed = try!(self.sign_call(request)); let r = take_weak!(self.client).call(&signed, analytics); if let Ok(executed) = r { // TODO maybe add other stuff to this? let mut ret = map!["output".to_owned() => to_value(&Bytes(executed.output)).unwrap()]; if let Some(trace) = executed.trace { ret.insert("trace".to_owned(), to_value(&Trace::from(trace)).unwrap()); } if let Some(vm_trace) = executed.vm_trace { ret.insert("vmTrace".to_owned(), vm_trace_to_object(&vm_trace)); } if let Some(state_diff) = executed.state_diff { ret.insert("stateDiff".to_owned(), state_diff_to_object(&state_diff)); } return Ok(Value::Object(ret)) } Ok(Value::Null) }) } }