/* 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 main.cpp * @author Gav Wood * @date 2014 * Ethereum client. */ #include #undef OK #include #include #include #include #include "Defaults.h" #include "Client.h" #include "PeerNetwork.h" #include "BlockChain.h" #include "State.h" #include "FileSystem.h" #include "Instruction.h" #include "BuildInfo.h" using namespace std; using namespace eth; using eth::Instruction; using eth::c_instructionInfo; bool isTrue(std::string const& _m) { return _m == "on" || _m == "yes" || _m == "true" || _m == "1"; } bool isFalse(std::string const& _m) { return _m == "off" || _m == "no" || _m == "false" || _m == "0"; } void help() { cout << "Usage eth [OPTIONS] " << endl << "Options:" << endl << " -a,--address Set the coinbase (mining payout) address to addr (default: auto)." << endl << " -c,--client-name Add a name to your client's version string (default: blank)." << endl << " -d,--db-path Load database from path (default: ~/.ethereum " << endl << " /Etherum or Library/Application Support/Ethereum)." << endl << " -h,--help Show this help message and exit." << endl << " -i,--interactive Enter interactive mode (default: non-interactive)." << endl << " -l,--listen Listen on the given port for incoming connected (default: 30303)." << endl << " -m,--mining Enable mining, optionally for a specified number of blocks (Default: off)" << endl << " -n,--upnp Use upnp for NAT (default: on)." << endl << " -o,--mode Start a full node or a peer node (Default: full)." << endl << " -p,--port Connect to remote port (default: 30303)." << endl << " -r,--remote Connect to remote host (default: none)." << endl << " -s,--secret Set the secret key for use with send command (default: auto)." << endl << " -u,--public-ip Force public ip to given (default; auto)." << endl << " -v,--verbosity <0 - 9> Set the log verbosity from 0 to 9 (Default: 8)." << endl << " -x,--peers Attempt to connect to given number of peers (Default: 5)." << endl << " -V,--version Show the version and exit." << endl; exit(0); } void interactiveHelp() { cout << "Commands:" << endl << " netstart Starts the network sybsystem on a specific port." << endl << " netstop Stops the network subsystem." << endl << " connect Connects to a specific peer." << endl << " minestart Starts mining." << endl << " minestop Stops mining." << endl << " address Gives the current address." << endl << " secret Gives the current secret" << endl << " block Gives the current block height." << endl << " balance Gives the current balance." << endl << " peers List the peers that are connected" << endl << " transact Executes a given transaction." << endl << " send Executes a given transaction with current secret." << endl << " inspect Dumps a contract to /.evm." << endl << " exit Exits the application." << endl; } string credits(bool _interactive = false) { std::ostringstream ccout; ccout << "Ethereum (++) " << ETH_QUOTED(ETH_VERSION) << endl << " Code by Gav Wood, (c) 2013, 2014." << endl << " Based on a design by Vitalik Buterin." << endl << endl; if (_interactive) { string vs = toString(ETH_QUOTED(ETH_VERSION)); vs = vs.substr(vs.find_first_of('.') + 1)[0]; int pocnumber = stoi(vs); string m_servers; if (pocnumber == 3) m_servers = "54.201.28.117"; if (pocnumber == 4) m_servers = "54.72.31.55"; ccout << "Type 'netstart 30303' to start networking" << endl; ccout << "Type 'connect " << m_servers << " 30303' to connect" << endl; ccout << "Type 'exit' to quit" << endl << endl; } return ccout.str(); } void version() { cout << "eth version " << ETH_QUOTED(ETH_VERSION) << endl; cout << "Build: " << ETH_QUOTED(ETH_BUILD_PLATFORM) << "/" << ETH_QUOTED(ETH_BUILD_TYPE) << endl; exit(0); } namespace nc { class nc_window_streambuf: public std::streambuf { public: nc_window_streambuf(WINDOW* p, std::ostream& os, unsigned long cursesAttr = 0); nc_window_streambuf(WINDOW* p, unsigned long _cursesAttr = 0); nc_window_streambuf(nc_window_streambuf const& _rhs); virtual ~nc_window_streambuf(); nc_window_streambuf& operator=(nc_window_streambuf const& _rhs); virtual int overflow(int c); virtual int sync(); private: void copy(nc_window_streambuf const& _rhs); WINDOW* m_pnl; unsigned long m_flags; std::ostream* m_os; std::streambuf* m_old; }; nc_window_streambuf::nc_window_streambuf(WINDOW * p, unsigned long _cursesAttr): m_pnl(p), m_flags(_cursesAttr), m_os(0), m_old(0) { // Tell parent class that we want to call overflow() for each // input char: setp(0, 0); setg(0, 0, 0); scrollok(p, true); mvwinch(p, 0, 0); } nc_window_streambuf::nc_window_streambuf(WINDOW* _p, std::ostream& _os, unsigned long _cursesAttr): m_pnl(_p), m_flags(_cursesAttr), m_os(&_os), m_old(_os.rdbuf()) { setp(0, 0); setg(0, 0, 0); _os.rdbuf(this); scrollok(_p, true); mvwinch(_p, 0, 0); } void nc_window_streambuf::copy(nc_window_streambuf const& _rhs) { if (this != &_rhs) { m_pnl = _rhs.m_pnl; m_flags = _rhs.m_flags; m_os = _rhs.m_os; m_old = _rhs.m_old; } } nc_window_streambuf::nc_window_streambuf(nc_window_streambuf const& _rhs): std::streambuf() { copy(_rhs); } nc_window_streambuf& nc_window_streambuf::operator=(nc_window_streambuf const& _rhs) { copy(_rhs); return *this; } nc_window_streambuf::~nc_window_streambuf() { if (m_os) m_os->rdbuf(m_old); } int nc_window_streambuf::overflow(int c) { int ret = c; if (c != EOF) { int x = 0; int y = 0; int mx = 0; int my = 0; getyx(m_pnl, y, x); getmaxyx(m_pnl, my, mx); if (y < 1) y = 1; if (x < 2) x = 2; if (x > mx - 4) { y++; x = 2; } if (m_flags) { wattron(m_pnl, m_flags); if (mvwaddch(m_pnl, y, x++, (chtype)c) == ERR) ret = EOF; wattroff(m_pnl, m_flags); } else if (mvwaddch(m_pnl, y, x++, (chtype)c) == ERR) ret = EOF; } if (c == EOF) // || std::isspace(c) if (sync() == EOF) ret = EOF; return ret; } int nc_window_streambuf::sync() { if (stdscr && m_pnl) return (wrefresh(m_pnl) == ERR) ? EOF : 0; return EOF; } } int main(int argc, char** argv) { unsigned short listenPort = 30303; string remoteHost; unsigned short remotePort = 30303; bool interactive = false; string dbPath; eth::uint mining = ~(eth::uint)0; NodeMode mode = NodeMode::Full; unsigned peers = 5; string publicIP; bool upnp = true; string clientName; // Init defaults Defaults::get(); // Our address. KeyPair us = KeyPair::create(); Address coinbase = us.address(); string configFile = getDataDir() + "/config.rlp"; bytes b = contents(configFile); if (b.size()) { RLP config(b); us = KeyPair(config[0].toHash()); coinbase = config[1].toHash
(); } else { RLPStream config(2); config << us.secret() << coinbase; writeFile(configFile, config.out()); } for (int i = 1; i < argc; ++i) { string arg = argv[i]; if ((arg == "-l" || arg == "--listen" || arg == "--listen-port") && i + 1 < argc) listenPort = (short)atoi(argv[++i]); else if ((arg == "-u" || arg == "--public-ip" || arg == "--public") && i + 1 < argc) publicIP = argv[++i]; else if ((arg == "-r" || arg == "--remote") && i + 1 < argc) remoteHost = argv[++i]; else if ((arg == "-p" || arg == "--port") && i + 1 < argc) remotePort = (short)atoi(argv[++i]); else if ((arg == "-n" || arg == "--upnp") && i + 1 < argc) { string m = argv[++i]; if (isTrue(m)) upnp = true; else if (isFalse(m)) upnp = false; else { cerr << "Invalid UPnP option: " << m << endl; return -1; } } else if ((arg == "-c" || arg == "--client-name") && i + 1 < argc) clientName = argv[++i]; else if ((arg == "-a" || arg == "--address" || arg == "--coinbase-address") && i + 1 < argc) coinbase = h160(fromHex(argv[++i])); else if ((arg == "-s" || arg == "--secret") && i + 1 < argc) us = KeyPair(h256(fromHex(argv[++i]))); else if (arg == "-i" || arg == "--interactive") interactive = true; else if ((arg == "-d" || arg == "--path" || arg == "--db-path") && i + 1 < argc) dbPath = argv[++i]; else if ((arg == "-m" || arg == "--mining") && i + 1 < argc) { string m = argv[++i]; if (isTrue(m)) mining = ~(eth::uint)0; else if (isFalse(m)) mining = 0; else if (int i = stoi(m)) mining = i; else { cerr << "Unknown mining option: " << m << endl; return -1; } } else if ((arg == "-v" || arg == "--verbosity") && i + 1 < argc) g_logVerbosity = atoi(argv[++i]); else if ((arg == "-x" || arg == "--peers") && i + 1 < argc) peers = atoi(argv[++i]); else if ((arg == "-o" || arg == "--mode") && i + 1 < argc) { string m = argv[++i]; if (m == "full") mode = NodeMode::Full; else if (m == "peer") mode = NodeMode::PeerServer; else { cerr << "Unknown mode: " << m << endl; return -1; } } else if (arg == "-h" || arg == "--help") help(); else if (arg == "-V" || arg == "--version") version(); else remoteHost = argv[i]; } if (!clientName.empty()) clientName += "/"; Client c("Ethereum(++)/" + clientName + "v" ETH_QUOTED(ETH_VERSION) "/" ETH_QUOTED(ETH_BUILD_TYPE) "/" ETH_QUOTED(ETH_BUILD_PLATFORM), coinbase, dbPath); cout << credits(); if (interactive) { std::ostringstream ccout; /* Initialize ncurses */ const char* chr; char* str = new char[255]; int width; int height; int y = 0; int x = 2; string cmd; WINDOW * mainwin, * consolewin, * logwin, * blockswin, * pendingwin, * contractswin, * peerswin; if (!(mainwin = initscr())) { cerr << "Error initialising ncurses."; return -1; } getmaxyx(mainwin, height, width); int qwidth = width / 4 - 4; nonl(); nocbreak(); timeout(30000); echo(); keypad(mainwin, true); logwin = newwin(height * 2 / 5 - 2, width, height * 3 / 5, 0); nc::nc_window_streambuf outbuf(logwin, std::cout); // nc::nc_window_streambuf errbuf( logwin, std::cerr ); g_logVerbosity = 1; // Force verbosity level for now consolewin = newwin(height * 3 / 5, width / 4, 0, 0); nc::nc_window_streambuf coutbuf(consolewin, ccout); blockswin = newwin(height * 3 / 5, width / 4, 0, width / 4); pendingwin = newwin(height * 1 / 5, width / 4, 0, width * 2 / 4); peerswin = newwin(height * 2 / 5, width / 4, height * 1 / 5, width * 2 / 4); contractswin = newwin(height * 3 / 5, width / 4, 0, width * 3 / 4); int vl = height * 3 / 5 - 4; wsetscrreg(consolewin, 1, vl); wsetscrreg(blockswin, 1, vl); wsetscrreg(pendingwin, 1, vl); wsetscrreg(peerswin, 1, vl); wsetscrreg(contractswin, 1, vl); mvwprintw(mainwin, 1, x, "> "); wresize(mainwin, 3, width); mvwin(mainwin, height - 3, 0); wmove(mainwin, 1, 4); if (!remoteHost.empty()) c.startNetwork(listenPort, remoteHost, remotePort, mode, peers, publicIP, upnp); while (true) { wclrtobot(consolewin); wclrtobot(pendingwin); wclrtobot(peerswin); wclrtobot(contractswin); ccout << credits(true); // Prompt wmove(mainwin, 1, 4); getstr(str); string s(str); istringstream iss(s); iss >> cmd; // Address ccout << "Address:" << endl; chr = toHex(us.address().asArray()).c_str(); ccout << chr << endl << endl; mvwprintw(mainwin, 1, x, "> "); clrtoeol(); if (s.length() > 1) { ccout << "> "; ccout << str << endl; } if (cmd == "netstart") { eth::uint port; iss >> port; c.startNetwork((short)port); } else if (cmd == "connect") { string addr; eth::uint port; iss >> addr >> port; c.connect(addr, (short)port); } else if (cmd == "netstop") c.stopNetwork(); else if (cmd == "minestart") c.startMining(); else if (cmd == "minestop") c.stopMining(); else if (cmd == "address") { ccout << "Current address:" << endl; const char* addchr = toHex(us.address().asArray()).c_str(); ccout << addchr << endl; } else if (cmd == "secret") { ccout << "Current secret:" << endl; const char* addchr = toHex(us.secret().asArray()).c_str(); ccout << addchr << endl; } else if (cmd == "block") { eth::uint n = c.blockChain().details().number; ccout << "Current block # "; const char* addchr = toString(n).c_str(); ccout << addchr << endl; } else if (cmd == "peers") { for (auto it: c.peers()) cout << it.host << ":" << it.port << ", " << it.clientVersion << ", " << std::chrono::duration_cast(it.lastPing).count() << "ms" << endl; } else if (cmd == "balance") { u256 balance = c.state().balance(us.address()); ccout << "Current balance:" << endl; const char* addchr = toString(balance).c_str(); ccout << addchr << endl; } else if (cmd == "transact") { string sechex; string rechex; u256 amount; u256 gasPrice; u256 gas; iss >> sechex >> rechex >> amount >> gasPrice >> gas; Secret secret = h256(fromHex(sechex)); Address dest = h160(fromHex(rechex)); bytes data; c.transact(secret, amount, gasPrice, gas, dest, data); } else if (cmd == "send") { string rechex; u256 amount; u256 gasPrice; u256 gas; iss >> rechex >> amount >> gasPrice >> gas; Address dest = h160(fromHex(rechex)); c.transact(us.secret(), amount, gasPrice, gas, dest, bytes()); } else if (cmd == "inspect") { string rechex; iss >> rechex; if (rechex.length() != 40) cout << "Invalid address length" << endl; else { c.lock(); auto h = h160(fromHex(rechex)); stringstream s; auto mem = c.state().contractMemory(h); u256 next = 0; unsigned numerics = 0; bool unexpectedNumeric = false; for (auto const& i: mem) { if (next < i.first) { unsigned j; for (j = 0; j <= numerics && next + j < i.first; ++j) s << (j < numerics || unexpectedNumeric ? " 0" : " STOP"); unexpectedNumeric = false; numerics -= min(numerics, j); if (next + j < i.first) s << "\n@" << showbase << hex << i.first << " "; } else if (!next) s << "@" << showbase << hex << i.first << " "; auto iit = c_instructionInfo.find((Instruction)(unsigned)i.second); if (numerics || iit == c_instructionInfo.end() || (u256)(unsigned)iit->first != i.second) // not an instruction or expecting an argument... { if (numerics) numerics--; else unexpectedNumeric = true; s << " " << showbase << hex << i.second; } else { auto const& ii = iit->second; s << " " << ii.name; numerics = ii.additional; } next = i.first + 1; } string outFile = getDataDir() + "/" + rechex + ".evm"; ofstream ofs; ofs.open(outFile, ofstream::binary); ofs.write(s.str().c_str(), s.str().length()); ofs.close(); c.unlock(); } } else if (cmd == "help") interactiveHelp(); else if (cmd == "exit") break; // Clear cmd at each pass cmd = ""; // Blocks auto const& st = c.state(); auto const& bc = c.blockChain(); y = 0; for (auto h = bc.currentHash(); h != bc.genesisHash(); h = bc.details(h).parent) { auto d = bc.details(h); string s = "# " + std::to_string(d.number) + ' ' + toString(h); // .abridged(); y += 1; mvwaddnstr(blockswin, y, x, s.c_str(), qwidth); for (auto const& i: RLP(bc.block(h))[1]) { Transaction t(i.data()); string ss; ss = t.receiveAddress ? " " + toString(toHex(t.safeSender().asArray())) + " " + (st.isContractAddress(t.receiveAddress) ? '*' : '-') + "> " + toString(t.receiveAddress) + ": " + toString(formatBalance(t.value)) + " [" + toString((unsigned)t.nonce) + "]": " " + toString(toHex(t.safeSender().asArray())) + " +> " + toString(right160(t.sha3())) + ": " + toString(formatBalance(t.value)) + " [" + toString((unsigned)t.nonce) + "]"; y += 1; mvwaddnstr(blockswin, y, x, ss.c_str(), qwidth - 2); if (y > height * 3 / 5 - 2) break; } if (y > height * 3 / 5 - 2) break; } // Pending y = 0; for (Transaction const& t: c.pending()) { string ss; if (t.receiveAddress) ss = toString(toHex(t.safeSender().asArray())) + " " + (st.isContractAddress(t.receiveAddress) ? '*' : '-') + "> " + toString(t.receiveAddress) + ": " + toString(formatBalance(t.value)) + " " + " [" + toString((unsigned)t.nonce) + "]"; else ss = toString(toHex(t.safeSender().asArray())) + " +> " + toString(right160(t.sha3())) + ": " + toString(formatBalance(t.value)) + "[" + toString((unsigned)t.nonce) + "]"; y += 1; mvwaddnstr(pendingwin, y, x, ss.c_str(), qwidth); if (y > height * 3 / 5 - 4) break; } // Contracts auto acs = st.addresses(); y = 0; for (auto n = 0; n < 2; ++n) for (auto const& i: acs) { auto r = i.first; if (st.isContractAddress(r)) { string ss; ss = toString(r) + " : " + toString(formatBalance(i.second)) + " [" + toString((unsigned)st.transactionsFrom(i.first)) + "]"; y += 1; mvwaddnstr(contractswin, y, x, ss.c_str(), qwidth); if (y > height * 3 / 5 - 2) break; } } // Peers y = 0; string psc; string pss; auto cp = c.peers(); psc = toString(cp.size()) + " peer(s)"; for (PeerInfo const& i: cp) { pss = toString(chrono::duration_cast(i.lastPing).count()) + " ms - " + i.host + ":" + toString(i.port) + " - " + i.clientVersion; y += 1; mvwaddnstr(peerswin, y, x, pss.c_str(), qwidth); if (y > height * 2 / 5 - 4) break; } box(consolewin, 0, 0); box(blockswin, 0, 0); box(pendingwin, 0, 0); box(peerswin, 0, 0); box(contractswin, 0, 0); box(mainwin, 0, 0); // Balance mvwprintw(consolewin, 0, x, "Balance: "); u256 balance = c.state().balance(us.address()); chr = toString(balance).c_str(); mvwprintw(consolewin, 0, 11, chr); wmove(consolewin, 1, x); // Block mvwprintw(blockswin, 0, x, "Block # "); eth::uint n = c.blockChain().details().number; chr = toString(n).c_str(); mvwprintw(blockswin, 0, 10, chr); mvwprintw(pendingwin, 0, x, "Pending"); mvwprintw(contractswin, 0, x, "Contracts"); // Peers mvwprintw(peerswin, 0, x, "Peers: "); chr = toString(c.peers().size()).c_str(); mvwprintw(peerswin, 0, 9, chr); wrefresh(consolewin); wrefresh(blockswin); wrefresh(pendingwin); wrefresh(peerswin); wrefresh(contractswin); wrefresh(mainwin); } delwin(contractswin); delwin(peerswin); delwin(pendingwin); delwin(blockswin); delwin(consolewin); delwin(logwin); delwin(mainwin); endwin(); refresh(); } else { cout << "Address: " << endl << toHex(us.address().asArray()) << endl; c.startNetwork(listenPort, remoteHost, remotePort, mode, peers, publicIP, upnp); eth::uint n = c.blockChain().details().number; if (mining) c.startMining(); while (true) { if (c.blockChain().details().number - n == mining) c.stopMining(); this_thread::sleep_for(chrono::milliseconds(100)); } } return 0; }