/* This file is part of cpp-ethereum. cpp-ethereum 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. cpp-ethereum 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 cpp-ethereum. If not, see . */ /** @file WebThreeStubServerBase.cpp * @authors: * Gav Wood * Marek Kotewicz * @date 2014 */ // Make sure boost/asio.hpp is included before windows.h. #include #include #include #include #include #include #include #include #include #include #include #include #include #ifndef _MSC_VER #include #endif #include "WebThreeStubServerBase.h" #include "AccountHolder.h" using namespace std; using namespace jsonrpc; using namespace dev; using namespace dev::eth; static Json::Value toJson(dev::eth::BlockInfo const& _bi) { Json::Value res; res["hash"] = boost::lexical_cast(_bi.hash); res["parentHash"] = toJS(_bi.parentHash); res["sha3Uncles"] = toJS(_bi.sha3Uncles); res["miner"] = toJS(_bi.coinbaseAddress); res["stateRoot"] = toJS(_bi.stateRoot); res["transactionsRoot"] = toJS(_bi.transactionsRoot); res["difficulty"] = toJS(_bi.difficulty); res["number"] = toJS(_bi.number); res["gasLimit"] = toJS(_bi.gasLimit); res["timestamp"] = toJS(_bi.timestamp); res["extraData"] = toJS(_bi.extraData); res["nonce"] = toJS(_bi.nonce); return res; } static Json::Value toJson(dev::eth::Transaction const& _t) { Json::Value res; res["hash"] = toJS(_t.sha3()); res["input"] = toJS(_t.data()); res["to"] = toJS(_t.receiveAddress()); res["from"] = toJS(_t.safeSender()); res["gas"] = toJS(_t.gas()); res["gasPrice"] = toJS(_t.gasPrice()); res["nonce"] = toJS(_t.nonce()); res["value"] = toJS(_t.value()); return res; } static Json::Value toJson(dev::eth::BlockInfo const& _bi, Transactions const& _ts) { Json::Value res = toJson(_bi); res["transactions"] = Json::Value(Json::arrayValue); for (Transaction const& t: _ts) res["transactions"].append(toJson(t)); return res; } static Json::Value toJson(dev::eth::BlockInfo const& _bi, TransactionHashes const& _ts) { Json::Value res = toJson(_bi); res["transactions"] = Json::Value(Json::arrayValue); for (h256 const& t: _ts) res["transactions"].append(toJS(t)); return res; } static Json::Value toJson(dev::eth::TransactionSkeleton const& _t) { Json::Value res; res["to"] = toJS(_t.to); res["from"] = toJS(_t.from); res["gas"] = toJS(_t.gas); res["gasPrice"] = toJS(_t.gasPrice); res["value"] = toJS(_t.value); res["data"] = toJS(_t.data, 32); return res; } static Json::Value toJson(dev::eth::LocalisedLogEntry const& _e) { Json::Value res; if (_e.transactionHash) { res["data"] = toJS(_e.data); res["address"] = toJS(_e.address); res["topics"] = Json::Value(Json::arrayValue); for (auto const& t: _e.topics) res["topics"].append(toJS(t)); res["number"] = _e.number; res["hash"] = toJS(_e.transactionHash); } return res; } static Json::Value toJson(dev::eth::LocalisedLogEntries const& _es) { Json::Value res(Json::arrayValue); for (dev::eth::LocalisedLogEntry const& e: _es) res.append(toJson(e)); return res; } static Json::Value toJson(map const& _storage) { Json::Value res(Json::objectValue); for (auto i: _storage) res[toJS(i.first)] = toJS(i.second); return res; } static unsigned toBlockNumber(std::string const& _js) { if (_js == "latest") return LatestBlock; else if (_js == "earliest") return 0; else if (_js == "pending") return PendingBlock; else return (unsigned)jsToInt(_js); } static dev::eth::LogFilter toLogFilter(Json::Value const& _json) // commented to avoid warning. Uncomment once in use @ PoC-7. { dev::eth::LogFilter filter; if (!_json.isObject() || _json.empty()) return filter; // check only !empty. it should throw exceptions if input params are incorrect if (!_json["fromBlock"].empty()) filter.withEarliest(toBlockNumber(_json["fromBlock"].asString())); if (!_json["toBlock"].empty()) filter.withLatest(toBlockNumber(_json["toBlock"].asString())); if (!_json["address"].empty()) { if (_json["address"].isArray()) for (auto i : _json["address"]) filter.address(jsToAddress(i.asString())); else filter.address(jsToAddress(_json["address"].asString())); } if (!_json["topics"].empty()) for (unsigned i = 0; i < _json["topics"].size(); i++) filter.topic(i, jsToFixed<32>(_json["topics"][i].asString())); return filter; } static shh::Message toMessage(Json::Value const& _json) { shh::Message ret; if (!_json["from"].empty()) ret.setFrom(jsToPublic(_json["from"].asString())); if (!_json["to"].empty()) ret.setTo(jsToPublic(_json["to"].asString())); if (!_json["payload"].empty()) ret.setPayload(jsToBytes(_json["payload"].asString())); return ret; } static shh::Envelope toSealed(Json::Value const& _json, shh::Message const& _m, Secret _from) { unsigned ttl = 50; unsigned workToProve = 50; shh::BuildTopic bt; if (!_json["ttl"].empty()) ttl = jsToInt(_json["ttl"].asString()); if (!_json["workToProve"].empty()) workToProve = jsToInt(_json["workToProve"].asString()); if (!_json["topics"].empty()) for (auto i: _json["topics"]) bt.shift(jsToBytes(i.asString())); return _m.seal(_from, bt, ttl, workToProve); } static pair toWatch(Json::Value const& _json) { shh::BuildTopic bt; Public to; if (!_json["to"].empty()) to = jsToPublic(_json["to"].asString()); if (!_json["topics"].empty()) for (auto i: _json["topics"]) bt.shift(jsToBytes(i.asString())); return make_pair(bt, to); } static Json::Value toJson(h256 const& _h, shh::Envelope const& _e, shh::Message const& _m) { Json::Value res; res["hash"] = toJS(_h); res["expiry"] = toJS(_e.expiry()); res["sent"] = toJS(_e.sent()); res["ttl"] = toJS(_e.ttl()); res["workProved"] = toJS(_e.workProved()); res["topics"] = Json::Value(Json::arrayValue); for (auto const& t: _e.topic()) res["topics"].append(toJS(t)); res["payload"] = toJS(_m.payload()); res["from"] = toJS(_m.from()); res["to"] = toJS(_m.to()); return res; } WebThreeStubServerBase::WebThreeStubServerBase(AbstractServerConnector& _conn, vector const& _accounts): AbstractWebThreeStubServer(_conn), m_accounts(make_shared(bind(&WebThreeStubServerBase::client, this))) { m_accounts->setAccounts(_accounts); } void WebThreeStubServerBase::setIdentities(vector const& _ids) { m_ids.clear(); for (auto i: _ids) m_ids[i.pub()] = i.secret(); } string WebThreeStubServerBase::web3_sha3(string const& _param1) { return toJS(sha3(jsToBytes(_param1))); } string WebThreeStubServerBase::net_peerCount() { return toJS(network()->peerCount()); } bool WebThreeStubServerBase::net_listening() { return network()->isNetworkStarted(); } string WebThreeStubServerBase::eth_coinbase() { return toJS(client()->address()); } bool WebThreeStubServerBase::eth_mining() { return client()->isMining(); } string WebThreeStubServerBase::eth_gasPrice() { return toJS(10 * dev::eth::szabo); } Json::Value WebThreeStubServerBase::eth_accounts() { Json::Value ret(Json::arrayValue); for (auto const& i: m_accounts->getAllAccounts()) ret.append(toJS(i)); return ret; } string WebThreeStubServerBase::eth_blockNumber() { return toJS(client()->number()); } string WebThreeStubServerBase::eth_getBalance(string const& _address, string const& _blockNumber) { try { return toJS(client()->balanceAt(jsToAddress(_address), toBlockNumber(_blockNumber))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } string WebThreeStubServerBase::eth_getStorageAt(string const& _address, string const& _position, string const& _blockNumber) { try { return toJS(client()->stateAt(jsToAddress(_address), jsToU256(_position), toBlockNumber(_blockNumber))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } string WebThreeStubServerBase::eth_getTransactionCount(string const& _address, string const& _blockNumber) { try { return toJS(client()->countAt(jsToAddress(_address), toBlockNumber(_blockNumber))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } string WebThreeStubServerBase::eth_getBlockTransactionCountByHash(string const& _blockHash) { try { return toJS(client()->transactionCount(jsToFixed<32>(_blockHash))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } string WebThreeStubServerBase::eth_getBlockTransactionCountByNumber(string const& _blockNumber) { try { return toJS(client()->transactionCount(client()->hashFromNumber(toBlockNumber(_blockNumber)))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } string WebThreeStubServerBase::eth_getUncleCountByBlockHash(string const& _blockHash) { try { return toJS(client()->uncleCount(jsToFixed<32>(_blockHash))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } string WebThreeStubServerBase::eth_getUncleCountByBlockNumber(string const& _blockNumber) { try { return toJS(client()->uncleCount(client()->hashFromNumber(toBlockNumber(_blockNumber)))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } string WebThreeStubServerBase::eth_getCode(string const& _address, string const& _blockNumber) { try { return toJS(client()->codeAt(jsToAddress(_address), toBlockNumber(_blockNumber))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } static TransactionSkeleton toTransaction(Json::Value const& _json) { TransactionSkeleton ret; if (!_json.isObject() || _json.empty()) return ret; if (!_json["from"].empty()) ret.from = jsToAddress(_json["from"].asString()); if (!_json["to"].empty()) ret.to = jsToAddress(_json["to"].asString()); else ret.creation = true; if (!_json["value"].empty()) ret.value = jsToU256(_json["value"].asString()); if (!_json["gas"].empty()) ret.gas = jsToU256(_json["gas"].asString()); if (!_json["gasPrice"].empty()) ret.gasPrice = jsToU256(_json["gasPrice"].asString()); if (!_json["data"].empty()) // ethereum.js has preconstructed the data array ret.data = jsToBytes(_json["data"].asString()); if (!_json["code"].empty()) ret.data = jsToBytes(_json["code"].asString()); return ret; } string WebThreeStubServerBase::eth_sendTransaction(Json::Value const& _json) { TransactionSkeleton t; try { t = toTransaction(_json); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } string ret; if (!t.from) t.from = m_accounts->getDefaultTransactAccount(); if (t.creation) ret = toJS(right160(sha3(rlpList(t.from, client()->countAt(t.from)))));; if (!t.gasPrice) t.gasPrice = 10 * dev::eth::szabo; // TODO: should be determined by user somehow. if (!t.gas) t.gas = min(client()->gasLimitRemaining(), client()->balanceAt(t.from) / t.gasPrice); if (m_accounts->isRealAccount(t.from)) authenticate(t, false); else if (m_accounts->isProxyAccount(t.from)) authenticate(t, true); return ret; } string WebThreeStubServerBase::eth_call(Json::Value const& _json, string const& _blockNumber) { TransactionSkeleton t; int number; try { t = toTransaction(_json); number = toBlockNumber(_blockNumber); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } string ret; if (!t.from) t.from = m_accounts->getDefaultTransactAccount(); if (!m_accounts->isRealAccount(t.from)) return ret; if (!t.gasPrice) t.gasPrice = 10 * dev::eth::szabo; if (!t.gas) t.gas = min(client()->gasLimitRemaining(), client()->balanceAt(t.from) / t.gasPrice); ret = toJS(client()->call(m_accounts->secretKey(t.from), t.value, t.to, t.data, t.gas, t.gasPrice, number).output); return ret; } bool WebThreeStubServerBase::eth_flush() { client()->flushTransactions(); return true; } Json::Value WebThreeStubServerBase::eth_getBlockByHash(string const& _blockHash, bool _includeTransactions) { try { auto h = jsToFixed<32>(_blockHash); if (_includeTransactions) return toJson(client()->blockInfo(h), client()->transactions(h)); else return toJson(client()->blockInfo(h), client()->transactionHashes(h)); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::eth_getBlockByNumber(string const& _blockNumber, bool _includeTransactions) { try { auto h = client()->hashFromNumber(jsToInt(_blockNumber)); if (_includeTransactions) return toJson(client()->blockInfo(h), client()->transactions(h)); else return toJson(client()->blockInfo(h), client()->transactionHashes(h)); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::eth_getTransactionByHash(string const& _transactionHash) { try { return toJson(client()->transaction(jsToFixed<32>(_transactionHash))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::eth_getTransactionByBlockHashAndIndex(string const& _blockHash, string const& _transactionIndex) { try { return toJson(client()->transaction(jsToFixed<32>(_blockHash), jsToInt(_transactionIndex))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::eth_getTransactionByBlockNumberAndIndex(string const& _blockNumber, string const& _transactionIndex) { try { return toJson(client()->transaction(client()->hashFromNumber(jsToInt(_blockNumber)), jsToInt(_transactionIndex))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::eth_getUncleByBlockHashAndIndex(string const& _blockHash, string const& _uncleIndex) { try { return toJson(client()->uncle(jsToFixed<32>(_blockHash), jsToInt(_uncleIndex))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::eth_getUncleByBlockNumberAndIndex(string const& _blockNumber, string const& _uncleIndex) { try { return toJson(client()->uncle(client()->hashFromNumber(toBlockNumber(_blockNumber)), jsToInt(_uncleIndex))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::eth_getCompilers() { Json::Value ret(Json::arrayValue); ret.append("lll"); ret.append("solidity"); #ifndef _MSC_VER ret.append("serpent"); #endif return ret; } string WebThreeStubServerBase::eth_compileLLL(string const& _code) { // TODO throw here jsonrpc errors string res; vector errors; res = toJS(dev::eth::compileLLL(_code, true, &errors)); cwarn << "LLL compilation errors: " << errors; return res; } string WebThreeStubServerBase::eth_compileSerpent(string const& _code) { // TODO throw here jsonrpc errors string res; #ifndef _MSC_VER try { res = toJS(dev::asBytes(::compile(_code))); } catch (string err) { cwarn << "Serpent compilation error: " << err; } catch (...) { cwarn << "Uncought serpent compilation exception"; } #endif return res; } string WebThreeStubServerBase::eth_compileSolidity(string const& _code) { // TOOD throw here jsonrpc errors string res; dev::solidity::CompilerStack compiler; try { res = toJS(compiler.compile(_code, true)); } catch (dev::Exception const& exception) { ostringstream error; solidity::SourceReferenceFormatter::printExceptionInformation(error, exception, "Error", compiler); cwarn << "Solidity compilation error: " << error.str(); } catch (...) { cwarn << "Uncought solidity compilation exception"; } return res; } string WebThreeStubServerBase::eth_newFilter(Json::Value const& _json) { try { return toJS(client()->installWatch(toLogFilter(_json))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } string WebThreeStubServerBase::eth_newBlockFilter(string const& _filter) { h256 filter; if (_filter.compare("chain") == 0 || _filter.compare("latest") == 0) filter = dev::eth::ChainChangedFilter; else if (_filter.compare("pending") == 0) filter = dev::eth::PendingChangedFilter; else BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); return toJS(client()->installWatch(filter)); } bool WebThreeStubServerBase::eth_uninstallFilter(string const& _filterId) { try { return client()->uninstallWatch(jsToInt(_filterId)); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::eth_getFilterChanges(string const& _filterId) { try { int id = jsToInt(_filterId); auto entries = client()->checkWatch(id); if (entries.size()) cnote << "FIRING WATCH" << id << entries.size(); return toJson(entries); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::eth_getFilterLogs(string const& _filterId) { try { return toJson(client()->logs(jsToInt(_filterId))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::eth_getLogs(Json::Value const& _json) { try { return toJson(client()->logs(toLogFilter(_json))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::eth_getWork() { Json::Value ret(Json::arrayValue); auto r = client()->getWork(); ret.append(toJS(r.first)); ret.append(toJS(r.second)); return ret; } bool WebThreeStubServerBase::eth_submitWork(string const& _nonce, string const& _mixHash) { try { return client()->submitWork(ProofOfWork::Proof{jsToFixed(_nonce), jsToFixed<32>(_mixHash)}); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } string WebThreeStubServerBase::eth_register(string const& _address) { try { return toJS(m_accounts->addProxyAccount(jsToAddress(_address))); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } bool WebThreeStubServerBase::eth_unregister(string const& _accountId) { try { return m_accounts->removeProxyAccount(jsToInt(_accountId)); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::eth_fetchQueuedTransactions(string const& _accountId) { try { auto id = jsToInt(_accountId); Json::Value ret(Json::arrayValue); // TODO: throw an error on no account with given id for (TransactionSkeleton const& t: m_accounts->getQueuedTransactions(id)) ret.append(toJson(t)); m_accounts->clearQueue(id); return ret; } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } bool WebThreeStubServerBase::db_put(string const& _name, string const& _key, string const& _value) { db()->put(_name, _key,_value); return true; } string WebThreeStubServerBase::db_get(string const& _name, string const& _key) { return db()->get(_name, _key);; } bool WebThreeStubServerBase::shh_post(Json::Value const& _json) { try { shh::Message m = toMessage(_json); Secret from; if (m.from() && m_ids.count(m.from())) { cwarn << "Silently signing message from identity" << m.from().abridged() << ": User validation hook goes here."; // TODO: insert validification hook here. from = m_ids[m.from()]; } face()->inject(toSealed(_json, m, from)); return true; } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } string WebThreeStubServerBase::shh_newIdentity() { KeyPair kp = KeyPair::create(); m_ids[kp.pub()] = kp.secret(); return toJS(kp.pub()); } bool WebThreeStubServerBase::shh_hasIdentity(string const& _identity) { try { return m_ids.count(jsToPublic(_identity)) > 0; } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } string WebThreeStubServerBase::shh_newGroup(string const& _id, string const& _who) { (void)_id; (void)_who; return ""; } string WebThreeStubServerBase::shh_addToGroup(string const& _group, string const& _who) { (void)_group; (void)_who; return ""; } string WebThreeStubServerBase::shh_newFilter(Json::Value const& _json) { try { pair w = toWatch(_json); auto ret = face()->installWatch(w.first); m_shhWatches.insert(make_pair(ret, w.second)); return toJS(ret); } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } bool WebThreeStubServerBase::shh_uninstallFilter(string const& _filterId) { try { face()->uninstallWatch(jsToInt(_filterId)); return true; } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::shh_getFilterChanges(string const& _filterId) { try { Json::Value ret(Json::arrayValue); int id = jsToInt(_filterId); auto pub = m_shhWatches[id]; if (!pub || m_ids.count(pub)) for (h256 const& h: face()->checkWatch(id)) { auto e = face()->envelope(h); shh::Message m; if (pub) { cwarn << "Silently decrypting message from identity" << pub.abridged() << ": User validation hook goes here."; m = e.open(face()->fullTopic(id), m_ids[pub]); } else m = e.open(face()->fullTopic(id)); if (!m) continue; ret.append(toJson(h, e, m)); } return ret; } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } Json::Value WebThreeStubServerBase::shh_getMessages(string const& _filterId) { try { Json::Value ret(Json::arrayValue); int id = jsToInt(_filterId); auto pub = m_shhWatches[id]; if (!pub || m_ids.count(pub)) for (h256 const& h: face()->watchMessages(id)) { auto e = face()->envelope(h); shh::Message m; if (pub) { cwarn << "Silently decrypting message from identity" << pub.abridged() << ": User validation hook goes here."; m = e.open(face()->fullTopic(id), m_ids[pub]); } else m = e.open(face()->fullTopic(id)); if (!m) continue; ret.append(toJson(h, e, m)); } return ret; } catch (...) { BOOST_THROW_EXCEPTION(JsonRpcException(Errors::ERROR_RPC_INVALID_PARAMS)); } } void WebThreeStubServerBase::authenticate(TransactionSkeleton const& _t, bool _toProxy) { if (_toProxy) m_accounts->queueTransaction(_t); else if (_t.to) client()->submitTransaction(m_accounts->secretKey(_t.from), _t.value, _t.to, _t.data, _t.gas, _t.gasPrice); else client()->submitTransaction(m_accounts->secretKey(_t.from), _t.value, _t.data, _t.gas, _t.gasPrice); } void WebThreeStubServerBase::setAccounts(const vector& _accounts) { m_accounts->setAccounts(_accounts); }