/* 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 ClientModel.cpp * @author Yann yann@ethdev.com * @author Arkadiy Paronyan arkadiy@ethdev.com * @date 2015 * Ethereum IDE client. */ #include #include #include #include #include #include #include #include "AppContext.h" #include "DebuggingStateWrapper.h" #include "QContractDefinition.h" #include "QVariableDeclaration.h" #include "ContractCallDataEncoder.h" #include "CodeModel.h" #include "ClientModel.h" #include "QEther.h" #include "Web3Server.h" #include "ClientModel.h" using namespace dev; using namespace dev::eth; namespace dev { namespace mix { class RpcConnector: public jsonrpc::AbstractServerConnector { public: virtual bool StartListening() override { return true; } virtual bool StopListening() override { return true; } virtual bool SendResponse(std::string const& _response, void*) override { m_response = QString::fromStdString(_response); return true; } QString response() const { return m_response; } private: QString m_response; }; ClientModel::ClientModel(AppContext* _context): m_context(_context), m_running(false), m_rpcConnector(new RpcConnector()) { qRegisterMetaType("QBigInt*"); qRegisterMetaType("QEther*"); qRegisterMetaType("QVariableDefinition*"); qRegisterMetaType("QVariableDefinitionList*"); qRegisterMetaType>("QList"); qRegisterMetaType>("QList"); qRegisterMetaType("QVariableDeclaration*"); qRegisterMetaType("AssemblyDebuggerData"); connect(this, &ClientModel::dataAvailable, this, &ClientModel::showDebugger, Qt::QueuedConnection); m_client.reset(new MixClient()); m_web3Server.reset(new Web3Server(*m_rpcConnector.get(), std::vector { m_client->userAccount() }, m_client.get())); _context->appEngine()->rootContext()->setContextProperty("clientModel", this); } ClientModel::~ClientModel() { } QString ClientModel::apiCall(QString const& _message) { m_rpcConnector->OnRequest(_message.toStdString(), nullptr); return m_rpcConnector->response(); } QString ClientModel::contractAddress() const { QString address = QString::fromStdString(dev::toJS(m_client->lastContractAddress())); return address; } void ClientModel::debugDeployment() { executeSequence(std::vector(), 10000000 * ether); } void ClientModel::debugState(QVariantMap _state) { u256 balance = (qvariant_cast(_state.value("balance")))->toU256Wei(); QVariantList transactions = _state.value("transactions").toList(); std::vector transactionSequence; TransactionSettings constructorTr; for (auto const& t: transactions) { QVariantMap transaction = t.toMap(); QString functionId = transaction.value("functionId").toString(); u256 gas = (qvariant_cast(transaction.value("gas")))->toU256Wei(); u256 value = (qvariant_cast(transaction.value("value")))->toU256Wei(); u256 gasPrice = (qvariant_cast(transaction.value("gasPrice")))->toU256Wei(); QVariantMap params = transaction.value("parameters").toMap(); TransactionSettings transactionSettings(functionId, value, gas, gasPrice); for (auto p = params.cbegin(); p != params.cend(); ++p) { QBigInt* param = qvariant_cast(p.value()); transactionSettings.parameterValues.insert(std::make_pair(p.key(), boost::get(param->internalValue()))); } if (transaction.value("executeConstructor").toBool()) constructorTr = transactionSettings; else transactionSequence.push_back(transactionSettings); } executeSequence(transactionSequence, balance, constructorTr); } void ClientModel::executeSequence(std::vector const& _sequence, u256 _balance, TransactionSettings const& ctrTransaction) { if (m_running) throw (std::logic_error("debugging already running")); auto compilerRes = m_context->codeModel()->code(); std::shared_ptr contractDef = compilerRes->sharedContract(); m_running = true; emit runStarted(); emit stateChanged(); //run sequence QtConcurrent::run([=]() { try { bytes contractCode = compilerRes->bytes(); std::vector transactonData; QFunctionDefinition* f = nullptr; ContractCallDataEncoder c; //encode data for all transactions for (auto const& t: _sequence) { f = nullptr; for (int tf = 0; tf < contractDef->functionsList().size(); tf++) { if (contractDef->functionsList().at(tf)->name() == t.functionId) { f = contractDef->functionsList().at(tf); break; } } if (!f) throw std::runtime_error("function " + t.functionId.toStdString() + " not found"); c.encode(f); for (int p = 0; p < f->parametersList().size(); p++) { QVariableDeclaration* var = (QVariableDeclaration*)f->parametersList().at(p); u256 value = 0; auto v = t.parameterValues.find(var->name()); if (v != t.parameterValues.cend()) value = v->second; c.encode(var, value); } transactonData.emplace_back(c.encodedData()); } //run contract creation first m_client->resetState(_balance); ExecutionResult debuggingContent = deployContract(contractCode, ctrTransaction); Address address = debuggingContent.contractAddress; for (unsigned i = 0; i < _sequence.size(); ++i) debuggingContent = callContract(address, transactonData.at(i), _sequence.at(i)); QList returnParameters; if (f) returnParameters = c.decode(f->returnParameters(), debuggingContent.returnValue); //we need to wrap states in a QObject before sending to QML. QList wStates; for (unsigned i = 0; i < debuggingContent.machineStates.size(); i++) { QPointer s(new DebuggingStateWrapper(debuggingContent.executionCode, debuggingContent.executionData.toBytes())); s->setState(debuggingContent.machineStates[i]); wStates.append(s); } //collect states for last transaction AssemblyDebuggerData code = DebuggingStateWrapper::getHumanReadableCode(debuggingContent.executionCode); emit dataAvailable(returnParameters, wStates, code); emit runComplete(); } catch(boost::exception const&) { emit runFailed(QString::fromStdString(boost::current_exception_diagnostic_information())); } catch(std::exception const& e) { emit runFailed(e.what()); } m_running = false; emit stateChanged(); }); } void ClientModel::showDebugger(QList const& _returnParam, QList const& _wStates, AssemblyDebuggerData const& _code) { m_context->appEngine()->rootContext()->setContextProperty("debugStates", QVariant::fromValue(_wStates)); m_context->appEngine()->rootContext()->setContextProperty("humanReadableExecutionCode", QVariant::fromValue(std::get<0>(_code))); m_context->appEngine()->rootContext()->setContextProperty("bytesCodeMapping", QVariant::fromValue(std::get<1>(_code))); m_context->appEngine()->rootContext()->setContextProperty("contractCallReturnParameters", QVariant::fromValue(new QVariableDefinitionList(_returnParam))); showDebuggerWindow(); } void ClientModel::showDebugError(QString const& _error) { //TODO: change that to a signal m_context->displayMessageDialog(tr("Debugger"), _error); } ExecutionResult ClientModel::deployContract(bytes const& _code, TransactionSettings const& _ctrTransaction) { Address newAddress; if (!_ctrTransaction.isEmpty()) newAddress = m_client->transact(m_client->userAccount().secret(), _ctrTransaction.value, _code, _ctrTransaction.gas, _ctrTransaction.gasPrice); else { u256 gasPrice = 10000000000000; u256 gas = 125000; u256 amount = 100; newAddress = m_client->transact(m_client->userAccount().secret(), amount, _code, gas, gasPrice); } Address lastAddress = m_client->lastContractAddress(); ExecutionResult r = m_client->lastExecutionResult(); if (newAddress != lastAddress) contractAddressChanged(); return r; } ExecutionResult ClientModel::callContract(Address const& _contract, bytes const& _data, TransactionSettings const& _tr) { m_client->transact(m_client->userAccount().secret(), _tr.value, _contract, _data, _tr.gas, _tr.gasPrice); ExecutionResult r = m_client->lastExecutionResult(); r.contractAddress = _contract; return r; } } }