/* 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 "Exceptions.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()), m_contractAddress(Address()) { qRegisterMetaType("QBigInt*"); qRegisterMetaType("QEther*"); qRegisterMetaType("QVariableDefinition*"); qRegisterMetaType("QVariableDefinitionList*"); qRegisterMetaType>("QList"); qRegisterMetaType>("QList"); qRegisterMetaType("QVariableDeclaration*"); qRegisterMetaType("QMachineState"); qRegisterMetaType("QInstruction"); qRegisterMetaType("QCode"); qRegisterMetaType("QCallData"); qRegisterMetaType("TransactionLogEntry"); connect(this, &ClientModel::runComplete, 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())); connect(m_web3Server.get(), &Web3Server::newTransaction, this, &ClientModel::onNewTransaction, Qt::DirectConnection); _context->appEngine()->rootContext()->setContextProperty("clientModel", this); } ClientModel::~ClientModel() { } QString ClientModel::apiCall(QString const& _message) { m_rpcConnector->OnRequest(_message.toStdString(), nullptr); return m_rpcConnector->response(); } void ClientModel::mine() { m_client->mine(); newBlock(); } QString ClientModel::contractAddress() const { return QString::fromStdString(dev::toJS(m_contractAddress)); } void ClientModel::debugDeployment() { executeSequence(std::vector(), 10000000 * ether); } void ClientModel::setupState(QVariantMap _state) { u256 balance = (qvariant_cast(_state.value("balance")))->toU256Wei(); QVariantList transactions = _state.value("transactions").toList(); std::vector transactionSequence; 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(); bool isStdContract = (transaction.value("stdContract").toBool()); if (isStdContract) { TransactionSettings transactionSettings(functionId, transaction.value("url").toString()); transactionSettings.gasPrice = 10000000000000; transactionSettings.gas = 125000; transactionSettings.value = 100; transactionSequence.push_back(transactionSettings); } else { 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()) transactionSettings.functionId.clear(); transactionSequence.push_back(transactionSettings); } } executeSequence(transactionSequence, balance); } void ClientModel::executeSequence(std::vector const& _sequence, u256 _balance) { if (m_running) BOOST_THROW_EXCEPTION(ExecutionStateException()); CompilationResult* compilerRes = m_context->codeModel()->code(); std::shared_ptr contractDef = compilerRes->sharedContract(); m_running = true; emit runStarted(); emit runStateChanged(); //run sequence QtConcurrent::run([=]() { try { bytes contractCode = compilerRes->bytes(); m_client->resetState(_balance); onStateReset(); for (TransactionSettings const& transaction: _sequence) { ContractCallDataEncoder encoder; QFunctionDefinition const* f = nullptr; if (!transaction.stdContractUrl.isEmpty()) { //std contract dev::bytes const& stdContractCode = m_context->codeModel()->getStdContractCode(transaction.functionId, transaction.stdContractUrl); Address address = deployContract(stdContractCode, transaction); m_stdContractAddresses[transaction.functionId] = address; m_stdContractNames[address] = transaction.functionId; } else { //encode data f = nullptr; if (transaction.functionId.isEmpty()) f = contractDef->constructor(); else for (QFunctionDefinition const* tf: contractDef->functionsList()) if (tf->name() == transaction.functionId) { f = tf; break; } if (!f) BOOST_THROW_EXCEPTION(FunctionNotFoundException() << FunctionName(transaction.functionId.toStdString())); encoder.encode(f); for (int p = 0; p < f->parametersList().size(); p++) { QVariableDeclaration* var = f->parametersList().at(p); u256 value = 0; auto v = transaction.parameterValues.find(var->name()); if (v != transaction.parameterValues.cend()) value = v->second; encoder.encode(var, value); } if (transaction.functionId.isEmpty()) { Address newAddress = deployContract(contractCode, transaction); if (newAddress != m_contractAddress) { m_contractAddress = newAddress; contractAddressChanged(); } } else callContract(m_contractAddress, encoder.encodedData(), transaction); } onNewTransaction(); } m_running = false; 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 runStateChanged(); }); } void ClientModel::showDebugger() { ExecutionResult const& last = m_client->record().back().transactions.back(); showDebuggerForTransaction(last); } void ClientModel::showDebuggerForTransaction(ExecutionResult const& _t) { //we need to wrap states in a QObject before sending to QML. QDebugData* debugData = new QDebugData(); QQmlEngine::setObjectOwnership(debugData, QQmlEngine::JavaScriptOwnership); QList codes; for (bytes const& code: _t.executionCode) codes.push_back(QMachineState::getHumanReadableCode(debugData, code)); QList data; for (bytes const& d: _t.transactionData) data.push_back(QMachineState::getDebugCallData(debugData, d)); QVariantList states; for (MachineState const& s: _t.machineStates) states.append(QVariant::fromValue(new QMachineState(debugData, s, codes[s.codeIndex], data[s.dataIndex]))); debugData->setStates(std::move(states)); //QList returnParameters; //returnParameters = encoder.decode(f->returnParameters(), debuggingContent.returnValue); //collect states for last transaction debugDataReady(debugData); } void ClientModel::debugTransaction(unsigned _block, unsigned _index) { auto const& t = m_client->record().at(_block).transactions.at(_index); showDebuggerForTransaction(t); } void ClientModel::showDebugError(QString const& _error) { //TODO: change that to a signal m_context->displayMessageDialog(tr("Debugger"), _error); } Address ClientModel::deployContract(bytes const& _code, TransactionSettings const& _ctrTransaction) { Address newAddress = m_client->transact(m_client->userAccount().secret(), _ctrTransaction.value, _code, _ctrTransaction.gas, _ctrTransaction.gasPrice); return newAddress; } void 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); } void ClientModel::onStateReset() { m_contractAddress = dev::Address(); m_stdContractAddresses.clear(); m_stdContractNames.clear(); emit stateCleared(); } void ClientModel::onNewTransaction() { unsigned block = m_client->number(); unsigned index = m_client->record().back().transactions.size() - 1; ExecutionResult const& tr = m_client->record().back().transactions.back(); QString address = QString::fromStdString(toJS(tr.address)); QString value = QString::fromStdString(dev::toString(tr.value)); QString contract = address; QString function; QString returned; bool creation = tr.contractAddress != 0; if (creation) returned = QString::fromStdString(toJS(tr.contractAddress)); else returned = QString::fromStdString(toJS(tr.returnValue)); //TODO: handle value transfer FixedHash<4> functionHash; bool call = false; if (creation) { //contract creation auto const stdContractName = m_stdContractNames.find(tr.contractAddress); if (stdContractName != m_stdContractNames.end()) { function = stdContractName->second; contract = function; } else function = QObject::tr("Constructor"); } else { //call if (tr.transactionData.size() > 0 && tr.transactionData.front().size() >= 4) { functionHash = FixedHash<4>(tr.transactionData.front().data(), FixedHash<4>::ConstructFromPointer); function = QString::fromStdString(toJS(functionHash)); call = true; } else function = QObject::tr(""); } if (m_contractAddress != 0 && (tr.address == m_contractAddress || tr.contractAddress == m_contractAddress)) { auto compilerRes = m_context->codeModel()->code(); QContractDefinition* def = compilerRes->contract(); contract = def->name(); if (call) { QFunctionDefinition* funcDef = def->getFunction(functionHash); if (funcDef) function = funcDef->name(); } } TransactionLogEntry* log = new TransactionLogEntry(block, index, contract, function, value, address, returned); QQmlEngine::setObjectOwnership(log, QQmlEngine::JavaScriptOwnership); emit newTransaction(log); } } }