/* 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #if ETH_JSCONSOLE || !ETH_TRUE #include #endif #if ETH_READLINE || !ETH_TRUE #include #include #endif #if ETH_JSONRPC || !ETH_TRUE #include #include #include #include #endif #include "BuildInfo.h" #if ETH_JSONRPC || !ETH_TRUE #include "PhoneHome.h" #include "Farm.h" #endif #include using namespace std; using namespace dev; using namespace dev::p2p; using namespace dev::eth; using namespace boost::algorithm; using dev::eth::Instruction; static std::atomic g_silence = {false}; void help() { cout << "Usage eth [OPTIONS]" << endl << "Options:" << endl << endl << "Operating mode (default is non-interactive node):" << endl #if ETH_JSCONSOLE || !ETH_TRUE << " console Enter interactive console mode (default: non-interactive)." << endl << " import Import file as a concatenated series of blocks." << endl << " export Export file as a concatenated series of blocks." << endl #endif << "Client mode (default):" << endl << " --olympic Use the Olympic (0.9) protocol." << endl << " --frontier Use the Frontier (1.0) protocol." << endl << " --private Use a private chain." << endl << " --genesis-json Import the genesis block information from the given json file." << endl << endl << " -o,--mode Start a full node or a peer node (default: full)." << endl << endl #if ETH_JSONRPC || !ETH_TRUE << " -j,--json-rpc Enable JSON-RPC server (default: off)." << endl << " --json-rpc-port Specify JSON-RPC server port (implies '-j', default: " << SensibleHttpPort << ")." << endl << " --admin Specify admin session key for JSON-RPC (default: auto-generated and printed at startup)." << endl #endif << " -K,--kill First kill the blockchain." << endl << " -R,--rebuild Rebuild the blockchain from the existing database." << endl << " --rescue Attempt to rescue a corrupt database." << endl << endl << " --import-presale Import a presale key; you'll need to type the password to this." << endl << " -s,--import-secret Import a secret key into the key store and use as the default." << endl << " -S,--import-session-secret Import a secret key into the key store and use as the default for this session only." << endl << " --sign-key
Sign all transactions with the key of the given address." << endl << " --session-sign-key
Sign all transactions with the key of the given address for this session only." << endl << " --master Give the master password for the key store." << endl << " --password Give a password for a private key." << endl << " --sentinel Set the sentinel for reporting bad blocks or chain issues." << endl << endl << "Client transacting:" << endl /*<< " -B,--block-fees Set the block fee profit in the reference unit e.g. ¢ (default: 15)." << endl << " -e,--ether-price Set the ether price in the reference unit e.g. ¢ (default: 30.679)." << endl << " -P,--priority <0 - 100> Default % priority of a transaction (default: 50)." << endl*/ << " --ask Set the minimum ask gas price under which no transactions will be mined (default " << toString(DefaultGasPrice) << " )." << endl << " --bid Set the bid gas price for to pay for transactions (default " << toString(DefaultGasPrice) << " )." << endl << endl << "Client mining:" << endl << " -a,--address Set the coinbase (mining payout) address to addr (default: auto)." << endl << " -m,--mining Enable mining, optionally for a specified number of blocks (default: off)" << endl << " -f,--force-mining Mine even when there are no transactions to mine (default: off)" << endl << " --mine-on-wrong-chain Mine even when we know it's the wrong chain (default: off)" << endl << " -C,--cpu When mining, use the CPU." << endl << " -G,--opencl When mining use the GPU via OpenCL." << endl << " --opencl-platform When mining using -G/--opencl use OpenCL platform n (default: 0)." << endl << " --opencl-device When mining using -G/--opencl use OpenCL device n (default: 0)." << endl << " -t, --mining-threads Limit number of CPU/GPU miners to n (default: use everything available on selected platform)" << endl << endl << "Client networking:" << endl << " --client-name Add a name to your client's version string (default: blank)." << endl << " --bootstrap Connect to the default Ethereum peerservers (default unless --no-discovery used)." << endl << " --no-bootstrap Do not connect to the default Ethereum peerservers (default only when --no-discovery is used)." << endl << " -x,--peers Attempt to connect to given number of peers (default: 11)." << endl << " --peer-stretch Accepted connection multiplier (default: 7)." << endl << " --public-ip Force advertised public ip to given (default: auto)." << endl << " --listen-ip (:) Listen on the given IP for incoming connections (default: 0.0.0.0)." << endl << " --listen Listen on the given port for incoming connections (default: 30303)." << endl << " -r,--remote (:) Connect to remote host (default: none)." << endl << " --port Connect to remote port (default: 30303)." << endl << " --network-id Only connect to other hosts with this network id." << endl << " --upnp Use UPnP for NAT (default: on)." << endl << " --peerset Space delimited list of type type:publickey@ipAddress[:port]" << endl << " Types:" << endl << " default Attempt connection when no other peers are available and pinning is disable." << endl << " require Keep connected at all times." << endl // TODO: // << " --trust-peers Space delimited list of publickeys." << endl << " --no-discovery Disable Node discovery, implies --no-bootstrap." << endl << " --pin Only accept or connect to trusted peers." << endl << " --hermit Equivalent to --no-discovery --pin." << endl << " --sociable Forces discovery and no pinning." << endl << endl; MinerCLI::streamHelp(cout); cout << "Client structured logging:" << endl << " --structured-logging Enable structured logging (default output to stdout)." << endl << " --structured-logging-format Set the structured logging time format." << endl << " --structured-logging-url Set the structured logging destination (currently only file:// supported)." << endl << endl << "Import/export modes:" << endl << " --from Export only from block n; n may be a decimal, a '0x' prefixed hash, or 'latest'." << endl << " --to Export only to block n (inclusive); n may be a decimal, a '0x' prefixed hash, or 'latest'." << endl << " --only Equivalent to --export-from n --export-to n." << endl << " --dont-check Avoids checking some of the aspects of blocks. Faster importing, but only do if you know the data is valid." << endl << endl << "General Options:" << endl << " -d,--db-path Load database from path (default: " << getDataDir() << ")" << endl #if ETH_EVMJIT || !ETH_TRUE << " --vm Select VM. Options are: interpreter, jit, smart. (default: interpreter)" << endl #endif << " -v,--verbosity <0 - 9> Set the log verbosity from 0 to 9 (default: 8)." << endl << " -V,--version Show the version and exit." << endl << " -h,--help Show this help message and exit." << endl << endl << "Experimental / Proof of Concept:" << endl << " --shh Enable Whisper" << endl << endl ; exit(0); } string ethCredits(bool _interactive = false) { std::ostringstream cout; if (_interactive) cout << "Type 'exit' to quit" << endl << endl; return credits() + cout.str(); } void version() { cout << "eth version " << dev::Version << endl; cout << "eth network protocol version: " << dev::eth::c_protocolVersion << endl; cout << "Client database version: " << dev::eth::c_databaseVersion << endl; cout << "Build: " << DEV_QUOTED(ETH_BUILD_PLATFORM) << "/" << DEV_QUOTED(ETH_BUILD_TYPE) << endl; exit(0); } void importPresale(KeyManager& _km, string const& _file, function _pass) { KeyPair k = _km.presaleSecret(contentsString(_file), [&](bool){ return _pass(); }); _km.import(k.secret(), "Presale wallet" + _file + " (insecure)"); } Address c_config = Address("ccdeac59d35627b7de09332e819d5159e7bb7250"); string pretty(h160 _a, dev::eth::State const& _st) { string ns; h256 n; if (h160 nameReg = (u160)_st.storage(c_config, 0)) n = _st.storage(nameReg, (u160)(_a)); if (n) { std::string s((char const*)n.data(), 32); if (s.find_first_of('\0') != string::npos) s.resize(s.find_first_of('\0')); ns = " " + s; } return ns; } inline bool isPrime(unsigned _number) { if (((!(_number & 1)) && _number != 2 ) || (_number < 2) || (_number % 3 == 0 && _number != 3)) return false; for(unsigned k = 1; 36 * k * k - 12 * k < _number; ++k) if ((_number % (6 * k + 1) == 0) || (_number % (6 * k - 1) == 0)) return false; return true; } enum class NodeMode { PeerServer, Full }; enum class OperationMode { Node, Import, Export }; enum class Format { Binary, Hex, Human }; void stopMiningAfterXBlocks(eth::Client* _c, unsigned _start, unsigned& io_mining) { if (io_mining != ~(unsigned)0 && io_mining && _c->isMining() && _c->blockChain().details().number - _start == io_mining) { _c->stopMining(); io_mining = ~(unsigned)0; } this_thread::sleep_for(chrono::milliseconds(100)); } int main(int argc, char** argv) { // Init defaults Defaults::get(); #if ETH_DEBUG g_logVerbosity = 4; #else g_logVerbosity = 1; #endif /// Operating mode. OperationMode mode = OperationMode::Node; string dbPath; // unsigned prime = 0; // bool yesIReallyKnowWhatImDoing = false; /// File name for import/export. string filename; bool safeImport = false; /// Hashes/numbers for export range. string exportFrom = "1"; string exportTo = "latest"; Format exportFormat = Format::Binary; /// General params for Node operation NodeMode nodeMode = NodeMode::Full; bool interactive = false; #if ETH_JSONRPC || !ETH_TRUE int jsonRPCURL = -1; #endif string jsonAdmin; string genesisJSON; dev::eth::Network releaseNetwork = c_network; u256 gasFloor = UndefinedU256; string privateChain; bool upnp = true; WithExisting withExisting = WithExisting::Trust; string sentinel; /// Networking params. string clientName; string listenIP; unsigned short listenPort = 30303; string publicIP; string remoteHost; unsigned short remotePort = 30303; unsigned peers = 11; unsigned peerStretch = 7; std::map> preferredNodes; bool bootstrap = true; bool disableDiscovery = false; bool pinning = false; bool enableDiscovery = false; bool noPinning = false; unsigned networkId = (unsigned)-1; /// Mining params unsigned mining = 0; bool forceMining = false; bool mineOnWrongChain = false; Address signingKey; Address sessionKey; Address beneficiary = signingKey; strings presaleImports; /// Structured logging params bool structuredLogging = false; string structuredLoggingFormat = "%Y-%m-%dT%H:%M:%S"; string structuredLoggingURL; /// Transaction params // TransactionPriority priority = TransactionPriority::Medium; // double etherPrice = 30.679; // double blockFees = 15.0; u256 askPrice = DefaultGasPrice; u256 bidPrice = DefaultGasPrice; // javascript console bool useConsole = false; /// Wallet password stuff string masterPassword; /// Whisper bool useWhisper = false; string configFile = getDataDir() + "/config.rlp"; bytes b = contents(configFile); strings passwordsToNote; Secrets toImport; if (b.size()) { try { RLP config(b); signingKey = config[0].toHash
(); beneficiary = config[1].toHash
(); } catch (...) {} } MinerCLI m(MinerCLI::OperationMode::None); for (int i = 1; i < argc; ++i) { string arg = argv[i]; if (m.interpretOption(i, argc, argv)) {} else if (arg == "--listen-ip" && i + 1 < argc) listenIP = argv[++i]; else if ((arg == "-l" || arg == "--listen" || arg == "--listen-port") && i + 1 < argc) { if (arg == "-l") cerr << "-l is DEPRECATED. It will be removed for the Frontier. Use --listen-port instead." << endl; listenPort = (short)atoi(argv[++i]); } else if ((arg == "-u" || arg == "--public-ip" || arg == "--public") && i + 1 < argc) { if (arg == "-u") cerr << "-u is DEPRECATED. It will be removed for the Frontier. Use --public-ip instead." << endl; publicIP = argv[++i]; } else if ((arg == "-r" || arg == "--remote") && i + 1 < argc) remoteHost = argv[++i]; else if ((arg == "-p" || arg == "--port") && i + 1 < argc) { if (arg == "-p") cerr << "-p is DEPRECATED. It will be removed for the Frontier. Use --port instead (or place directly as host:port)." << endl; remotePort = (short)atoi(argv[++i]); } else if (arg == "--password" && i + 1 < argc) passwordsToNote.push_back(argv[++i]); else if (arg == "--master" && i + 1 < argc) masterPassword = argv[++i]; else if ((arg == "-I" || arg == "--import" || arg == "import") && i + 1 < argc) { mode = OperationMode::Import; filename = argv[++i]; } else if (arg == "--dont-check") safeImport = true; else if ((arg == "-E" || arg == "--export" || arg == "export") && i + 1 < argc) { mode = OperationMode::Export; filename = argv[++i]; } /* else if (arg == "--prime" && i + 1 < argc) try { prime = stoi(argv[++i]); } catch (...) { cerr << "Bad " << arg << " option: " << argv[i] << endl; return -1; } else if (arg == "--yes-i-really-know-what-im-doing") yesIReallyKnowWhatImDoing = true; */ else if (arg == "--sentinel" && i + 1 < argc) sentinel = argv[++i]; else if (arg == "--mine-on-wrong-chain") mineOnWrongChain = true; else if (arg == "--format" && i + 1 < argc) { string m = argv[++i]; if (m == "binary") exportFormat = Format::Binary; else if (m == "hex") exportFormat = Format::Hex; else if (m == "human") exportFormat = Format::Human; else { cerr << "Bad " << arg << " option: " << m << endl; return -1; } } else if (arg == "--to" && i + 1 < argc) exportTo = argv[++i]; else if (arg == "--from" && i + 1 < argc) exportFrom = argv[++i]; else if (arg == "--only" && i + 1 < argc) exportTo = exportFrom = argv[++i]; else if ((arg == "-n" || arg == "-u" || arg == "--upnp") && i + 1 < argc) { if (arg == "-n") cerr << "-n is DEPRECATED. It will be removed for the Frontier. Use --upnp instead." << endl; string m = argv[++i]; if (isTrue(m)) upnp = true; else if (isFalse(m)) upnp = false; else { cerr << "Bad " << arg << " option: " << m << endl; return -1; } } else if (arg == "--network-id" && i + 1 < argc) try { networkId = stol(argv[++i]); } catch (...) { cerr << "Bad " << arg << " option: " << argv[i] << endl; return -1; } else if (arg == "--private" && i + 1 < argc) try { privateChain = argv[++i]; } catch (...) { cerr << "Bad " << arg << " option: " << argv[i] << endl; return -1; } else if (arg == "-K" || arg == "--kill-blockchain" || arg == "--kill") withExisting = WithExisting::Kill; else if (arg == "-R" || arg == "--rebuild") withExisting = WithExisting::Verify; else if (arg == "-R" || arg == "--rescue") withExisting = WithExisting::Rescue; else if ((arg == "-c" || arg == "--client-name") && i + 1 < argc) { if (arg == "-c") cerr << "-c is DEPRECATED. It will be removed for the Frontier. Use --client-name instead." << endl; clientName = argv[++i]; } else if ((arg == "-a" || arg == "--address" || arg == "--coinbase-address") && i + 1 < argc) try { beneficiary = h160(fromHex(argv[++i], WhenError::Throw)); } catch (BadHexCharacter&) { cerr << "Bad hex in " << arg << " option: " << argv[i] << endl; return -1; } catch (...) { cerr << "Bad " << arg << " option: " << argv[i] << endl; return -1; } else if ((arg == "-s" || arg == "--import-secret") && i + 1 < argc) { Secret s(fromHex(argv[++i])); toImport.emplace_back(s); signingKey = toAddress(s); } else if ((arg == "-S" || arg == "--import-session-secret") && i + 1 < argc) { Secret s(fromHex(argv[++i])); toImport.emplace_back(s); sessionKey = toAddress(s); } else if ((arg == "--sign-key") && i + 1 < argc) sessionKey = Address(fromHex(argv[++i])); else if ((arg == "--session-sign-key") && i + 1 < argc) sessionKey = Address(fromHex(argv[++i])); else if (arg == "--structured-logging-format" && i + 1 < argc) structuredLoggingFormat = string(argv[++i]); else if (arg == "--structured-logging") structuredLogging = true; else if (arg == "--structured-logging-url" && i + 1 < argc) { structuredLogging = true; structuredLoggingURL = argv[++i]; } else if ((arg == "-d" || arg == "--path" || arg == "--db-path") && i + 1 < argc) dbPath = argv[++i]; else if ((arg == "--genesis-json" || arg == "--genesis") && i + 1 < argc) { try { genesisJSON = contentsString(argv[++i]); } catch (...) { cerr << "Bad " << arg << " option: " << argv[i] << endl; return -1; } } else if (arg == "--frontier") releaseNetwork = eth::Network::Frontier; else if (arg == "--gas-floor" && i + 1 < argc) gasFloor = u256(argv[++i]); else if (arg == "--olympic") releaseNetwork = eth::Network::Olympic; /* else if ((arg == "-B" || arg == "--block-fees") && i + 1 < argc) { try { blockFees = stof(argv[++i]); } catch (...) { cerr << "Bad " << arg << " option: " << argv[i] << endl; return -1; } } else if ((arg == "-e" || arg == "--ether-price") && i + 1 < argc) { try { etherPrice = stof(argv[++i]); } catch (...) { cerr << "Bad " << arg << " option: " << argv[i] << endl; return -1; } }*/ else if (arg == "--ask" && i + 1 < argc) { try { askPrice = u256(argv[++i]); } catch (...) { cerr << "Bad " << arg << " option: " << argv[i] << endl; return -1; } } else if (arg == "--bid" && i + 1 < argc) { try { bidPrice = u256(argv[++i]); } catch (...) { cerr << "Bad " << arg << " option: " << argv[i] << endl; return -1; } } /* else if ((arg == "-P" || arg == "--priority") && i + 1 < argc) { string m = boost::to_lower_copy(string(argv[++i])); if (m == "lowest") priority = TransactionPriority::Lowest; else if (m == "low") priority = TransactionPriority::Low; else if (m == "medium" || m == "mid" || m == "default" || m == "normal") priority = TransactionPriority::Medium; else if (m == "high") priority = TransactionPriority::High; else if (m == "highest") priority = TransactionPriority::Highest; else try { priority = (TransactionPriority)(max(0, min(100, stoi(m))) * 8 / 100); } catch (...) { cerr << "Unknown " << arg << " option: " << m << endl; return -1; } }*/ else if ((arg == "-m" || arg == "--mining") && i + 1 < argc) { string m = argv[++i]; if (isTrue(m)) mining = ~(unsigned)0; else if (isFalse(m)) mining = 0; else try { mining = stoi(m); } catch (...) { cerr << "Unknown " << arg << " option: " << m << endl; return -1; } } else if (arg == "-b" || arg == "--bootstrap") bootstrap = true; else if (arg == "--no-bootstrap") bootstrap = false; else if (arg == "--no-discovery") { disableDiscovery = true; bootstrap = false; } else if (arg == "--pin") pinning = true; else if (arg == "--hermit") pinning = disableDiscovery = true; else if (arg == "--sociable") noPinning = enableDiscovery = true; else if (arg == "--import-presale" && i + 1 < argc) presaleImports.push_back(argv[++i]); else if (arg == "-f" || arg == "--force-mining") forceMining = true; else if (arg == "--old-interactive") interactive = true; #if ETH_JSONRPC || !ETH_TRUE else if ((arg == "-j" || arg == "--json-rpc")) jsonRPCURL = jsonRPCURL == -1 ? SensibleHttpPort : jsonRPCURL; else if (arg == "--json-rpc-port" && i + 1 < argc) jsonRPCURL = atoi(argv[++i]); else if (arg == "--json-admin" && i + 1 < argc) jsonAdmin = argv[++i]; #endif #if ETH_JSCONSOLE || !ETH_TRUE else if (arg == "-i" || arg == "--interactive" || arg == "--console" || arg == "console") useConsole = true; #endif 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 == "--peer-stretch" && i + 1 < argc) peerStretch = atoi(argv[++i]); else if (arg == "--peerset" && i + 1 < argc) { string peerset = argv[++i]; if (peerset.empty()) { cerr << "--peerset argument must not be empty"; return -1; } vector each; boost::split(each, peerset, boost::is_any_of("\t ")); for (auto const& p: each) { string type; string pubk; string hostIP; unsigned short port = c_defaultListenPort; // type:key@ip[:port] vector typeAndKeyAtHostAndPort; boost::split(typeAndKeyAtHostAndPort, p, boost::is_any_of(":")); if (typeAndKeyAtHostAndPort.size() < 2 || typeAndKeyAtHostAndPort.size() > 3) continue; type = typeAndKeyAtHostAndPort[0]; if (typeAndKeyAtHostAndPort.size() == 3) port = (uint16_t)atoi(typeAndKeyAtHostAndPort[2].c_str()); vector keyAndHost; boost::split(keyAndHost, typeAndKeyAtHostAndPort[1], boost::is_any_of("@")); if (keyAndHost.size() != 2) continue; pubk = keyAndHost[0]; if (pubk.size() != 40) continue; hostIP = keyAndHost[1]; // todo: use Network::resolveHost() if (hostIP.size() < 4 /* g.it */) continue; bool required = type == "required"; if (!required && type != "default") continue; Public publicKey(fromHex(pubk)); try { preferredNodes[publicKey] = make_pair(NodeIPEndpoint(bi::address::from_string(hostIP), port, port), required); } catch (...) { cerr << "Unrecognized peerset: " << peerset << endl; return -1; } } } else if ((arg == "-o" || arg == "--mode") && i + 1 < argc) { string m = argv[++i]; if (m == "full") nodeMode = NodeMode::Full; else if (m == "peer") nodeMode = NodeMode::PeerServer; else { cerr << "Unknown mode: " << m << endl; return -1; } } #if ETH_EVMJIT else if (arg == "--vm" && i + 1 < argc) { string vmKind = argv[++i]; if (vmKind == "interpreter") VMFactory::setKind(VMKind::Interpreter); else if (vmKind == "jit") VMFactory::setKind(VMKind::JIT); else if (vmKind == "smart") VMFactory::setKind(VMKind::Smart); else { cerr << "Unknown VM kind: " << vmKind << endl; return -1; } } #endif else if (arg == "--shh") useWhisper = true; else if (arg == "-h" || arg == "--help") help(); else if (arg == "-V" || arg == "--version") version(); else { cerr << "Invalid argument: " << arg << endl; exit(-1); } } // Set up all the chain config stuff. resetNetwork(releaseNetwork); if (!privateChain.empty()) { CanonBlockChain::forceGenesisExtraData(sha3(privateChain).asBytes()); CanonBlockChain::forceGenesisDifficulty(c_minimumDifficulty); CanonBlockChain::forceGenesisGasLimit(u256(1) << 32); } if (!genesisJSON.empty()) CanonBlockChain::setGenesis(genesisJSON); if (gasFloor != UndefinedU256) c_gasFloorTarget = gasFloor; if (networkId == (unsigned)-1) networkId = (unsigned)c_network; if (g_logVerbosity > 0) { cout << EthGrayBold "(++)Ethereum" EthReset << endl; if (c_network == eth::Network::Olympic) cout << "Welcome to Olympic!" << endl; else if (c_network == eth::Network::Frontier) cout << "Beware. You're entering the " EthMaroonBold "Frontier" EthReset "!" << endl; } m.execute(); KeyManager keyManager; for (auto const& s: passwordsToNote) keyManager.notePassword(s); writeFile(configFile, rlpList(signingKey, beneficiary)); if (sessionKey) signingKey = sessionKey; if (!clientName.empty()) clientName += "/"; string logbuf; std::string additional; if (interactive) g_logPost = [&](std::string const& a, char const*){ static SpinLock s_lock; SpinGuard l(s_lock); if (g_silence) logbuf += a + "\n"; else cout << "\r \r" << a << endl << additional << flush; // helpful to use OutputDebugString on windows #ifdef _WIN32 { OutputDebugStringA(a.data()); OutputDebugStringA("\n"); } #endif }; auto getPassword = [&](string const& prompt){ bool s = g_silence; g_silence = true; cout << endl; string ret = dev::getPassword(prompt); g_silence = s; return ret; }; auto getAccountPassword = [&](Address const& a){ return getPassword("Enter password for address " + keyManager.accountName(a) + " (" + a.abridged() + "; hint:" + keyManager.passwordHint(a) + "): "); }; StructuredLogger::get().initialize(structuredLogging, structuredLoggingFormat, structuredLoggingURL); auto netPrefs = publicIP.empty() ? NetworkPreferences(listenIP, listenPort, upnp) : NetworkPreferences(publicIP, listenIP ,listenPort, upnp); netPrefs.discovery = (privateChain.empty() && !disableDiscovery) || enableDiscovery; netPrefs.pin = (pinning || !privateChain.empty()) && !noPinning; auto nodesState = contents((dbPath.size() ? dbPath : getDataDir()) + "/network.rlp"); auto caps = useWhisper ? set{"eth", "shh"} : set{"eth"}; dev::WebThreeDirect web3( WebThreeDirect::composeClientVersion("++eth", clientName), dbPath, withExisting, nodeMode == NodeMode::Full ? caps : set(), netPrefs, &nodesState); web3.ethereum()->setMineOnBadChain(mineOnWrongChain); web3.ethereum()->setSentinel(sentinel); auto toNumber = [&](string const& s) -> unsigned { if (s == "latest") return web3.ethereum()->number(); if (s.size() == 64 || (s.size() == 66 && s.substr(0, 2) == "0x")) return web3.ethereum()->blockChain().number(h256(s)); try { return stol(s); } catch (...) { cerr << "Bad block number/hash option: " << s << endl; exit(-1); } }; if (mode == OperationMode::Export) { ofstream fout(filename, std::ofstream::binary); ostream& out = (filename.empty() || filename == "--") ? cout : fout; unsigned last = toNumber(exportTo); for (unsigned i = toNumber(exportFrom); i <= last; ++i) { bytes block = web3.ethereum()->blockChain().block(web3.ethereum()->blockChain().numberHash(i)); switch (exportFormat) { case Format::Binary: out.write((char const*)block.data(), block.size()); break; case Format::Hex: out << toHex(block) << endl; break; case Format::Human: out << RLP(block) << endl; break; default:; } } return 0; } if (mode == OperationMode::Import) { ifstream fin(filename, std::ifstream::binary); istream& in = (filename.empty() || filename == "--") ? cin : fin; unsigned alreadyHave = 0; unsigned good = 0; unsigned futureTime = 0; unsigned unknownParent = 0; unsigned bad = 0; chrono::steady_clock::time_point t = chrono::steady_clock::now(); double last = 0; unsigned lastImported = 0; unsigned imported = 0; while (in.peek() != -1) { bytes block(8); in.read((char*)block.data(), 8); block.resize(RLP(block, RLP::LaissezFaire).actualSize()); in.read((char*)block.data() + 8, block.size() - 8); switch (web3.ethereum()->queueBlock(block, safeImport)) { case ImportResult::Success: good++; break; case ImportResult::AlreadyKnown: alreadyHave++; break; case ImportResult::UnknownParent: unknownParent++; break; case ImportResult::FutureTimeUnknown: unknownParent++; futureTime++; break; case ImportResult::FutureTimeKnown: futureTime++; break; default: bad++; break; } // sync chain with queue tuple r = web3.ethereum()->syncQueue(10); imported += get<2>(r); double e = chrono::duration_cast(chrono::steady_clock::now() - t).count() / 1000.0; if ((unsigned)e >= last + 10) { auto i = imported - lastImported; auto d = e - last; cout << i << " more imported at " << (round(i * 10 / d) / 10) << " blocks/s. " << imported << " imported in " << e << " seconds at " << (round(imported * 10 / e) / 10) << " blocks/s (#" << web3.ethereum()->number() << ")" << endl; last = (unsigned)e; lastImported = imported; // cout << web3.ethereum()->blockQueueStatus() << endl; } } while (web3.ethereum()->blockQueue().items().first + web3.ethereum()->blockQueue().items().second > 0) { this_thread::sleep_for(chrono::seconds(1)); web3.ethereum()->syncQueue(100000); } double e = chrono::duration_cast(chrono::steady_clock::now() - t).count() / 1000.0; cout << imported << " imported in " << e << " seconds at " << (round(imported * 10 / e) / 10) << " blocks/s (#" << web3.ethereum()->number() << ")" << endl; return 0; } /* if (c_network == eth::Network::Frontier && !yesIReallyKnowWhatImDoing) { auto pd = contents(getDataDir() + "primes"); unordered_set primes = RLP(pd).toUnorderedSet(); while (true) { if (!prime) try { prime = stoi(getPassword("To enter the Frontier, enter a 6 digit prime that you have not entered before: ")); } catch (...) {} if (isPrime(prime) && !primes.count(prime)) break; prime = 0; } primes.insert(prime); writeFile(getDataDir() + "primes", rlp(primes)); } */ if (keyManager.exists()) { if (masterPassword.empty() || !keyManager.load(masterPassword)) while (true) { masterPassword = getPassword("Please enter your MASTER password: "); if (keyManager.load(masterPassword)) break; cout << "The password you entered is incorrect. If you have forgotten your password, and you wish to start afresh, manually remove the file: " + getDataDir("ethereum") + "/keys.info" << endl; } } else { while (masterPassword.empty()) { masterPassword = getPassword("Please enter a MASTER password to protect your key store (make it strong!): "); string confirm = getPassword("Please confirm the password by entering it again: "); if (masterPassword != confirm) { cout << "Passwords were different. Try again." << endl; masterPassword.clear(); } } keyManager.create(masterPassword); } for (auto const& presale: presaleImports) importPresale(keyManager, presale, [&](){ return getPassword("Enter your wallet password for " + presale + ": "); }); for (auto const& s: toImport) { keyManager.import(s, "Imported key (UNSAFE)"); if (!signingKey) signingKey = toAddress(s); } if (keyManager.accounts().empty()) { h128 uuid = keyManager.import(ICAP::createDirect(), "Default key"); if (!beneficiary) beneficiary = keyManager.address(uuid); if (!signingKey) signingKey = keyManager.address(uuid); writeFile(configFile, rlpList(signingKey, beneficiary)); } cout << ethCredits(); web3.setIdealPeerCount(peers); web3.setPeerStretch(peerStretch); // std::shared_ptr gasPricer = make_shared(u256(double(ether / 1000) / etherPrice), u256(blockFees * 1000)); std::shared_ptr gasPricer = make_shared(askPrice, bidPrice); eth::Client* c = nodeMode == NodeMode::Full ? web3.ethereum() : nullptr; StructuredLogger::starting(WebThreeDirect::composeClientVersion("++eth", clientName), dev::Version); if (c) { c->setGasPricer(gasPricer); c->setForceMining(forceMining); // TODO: expose sealant interface. c->setShouldPrecomputeDAG(m.shouldPrecompute()); c->setTurboMining(m.minerType() == MinerCLI::MinerType::GPU); c->setBeneficiary(beneficiary); c->setNetworkId(networkId); } auto renderFullAddress = [&](Address const& _a) -> std::string { return ICAP(_a).encoded() + " (" + toUUID(keyManager.uuid(_a)) + " - " + toHex(_a.ref().cropped(0, 4)) + ")"; }; cout << "Transaction Signer: " << renderFullAddress(signingKey) << endl; cout << "Mining Beneficiary: " << renderFullAddress(beneficiary) << endl; cout << "Foundation: " << renderFullAddress(Address("de0b295669a9fd93d5f28d9ec85e40f4cb697bae")) << endl; if (bootstrap || !remoteHost.empty() || disableDiscovery) { web3.startNetwork(); cout << "Node ID: " << web3.enode() << endl; } else cout << "Networking disabled. To start, use netstart or pass --bootstrap or a remote host." << endl; if (useConsole && jsonRPCURL == -1) jsonRPCURL = SensibleHttpPort; #if ETH_JSONRPC || !ETH_TRUE shared_ptr jsonrpcServer; unique_ptr jsonrpcConnector; if (jsonRPCURL > -1) { jsonrpcConnector = unique_ptr(new jsonrpc::HttpServer(jsonRPCURL, "", "", SensibleHttpThreads)); jsonrpcServer = make_shared(*jsonrpcConnector.get(), web3, make_shared([&](){ return web3.ethereum(); }, getAccountPassword, keyManager), vector(), keyManager, *gasPricer); jsonrpcServer->StartListening(); if (jsonAdmin.empty()) jsonAdmin = jsonrpcServer->newSession(SessionPermissions{{Privilege::Admin}}); else jsonrpcServer->addSession(jsonAdmin, SessionPermissions{{Privilege::Admin}}); cout << "JSONRPC Admin Session Key: " << jsonAdmin << endl; writeFile(getDataDir("web3") + "/session.key", jsonAdmin); writeFile(getDataDir("web3") + "/session.url", "http://localhost:" + toString(jsonRPCURL)); } #endif for (auto const& p: preferredNodes) if (p.second.second) web3.requirePeer(p.first, p.second.first); else web3.addNode(p.first, p.second.first); if (bootstrap) for (auto const& i: Host::pocHosts()) web3.requirePeer(i.first, i.second); if (!remoteHost.empty()) web3.addNode(p2p::NodeId(), remoteHost + ":" + toString(remotePort)); signal(SIGABRT, &Client::exitHandler); signal(SIGTERM, &Client::exitHandler); signal(SIGINT, &Client::exitHandler); if (c) { unsigned n = c->blockChain().details().number; if (mining) c->startMining(); if (useConsole) { #if ETH_JSCONSOLE || !ETH_TRUE JSLocalConsole console; shared_ptr rpcServer = make_shared(*console.connector(), web3, make_shared([&](){ return web3.ethereum(); }, getAccountPassword, keyManager), vector(), keyManager, *gasPricer); string sessionKey = rpcServer->newSession(SessionPermissions{{Privilege::Admin}}); console.eval("web3.admin.setSessionKey('" + sessionKey + "')"); while (!Client::shouldExit()) { console.readExpression(); stopMiningAfterXBlocks(c, n, mining); } rpcServer->StopListening(); #endif } else while (!Client::shouldExit()) stopMiningAfterXBlocks(c, n, mining); } else while (!Client::shouldExit()) this_thread::sleep_for(chrono::milliseconds(1000)); #if ETH_JSONRPC if (jsonrpcServer.get()) jsonrpcServer->StopListening(); #endif StructuredLogger::stopping(WebThreeDirect::composeClientVersion("++eth", clientName), dev::Version); auto netData = web3.saveNetwork(); if (!netData.empty()) writeFile((dbPath.size() ? dbPath : getDataDir()) + "/network.rlp", netData); return 0; }