/* 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 Client.h * @author Gav Wood * @date 2014 */ #pragma once #include #include #include #include #include #include #include #include #include #include "Guards.h" #include "BlockChain.h" #include "TransactionQueue.h" #include "State.h" #include "PeerNetwork.h" namespace eth { struct MineProgress { double requirement; double best; double current; uint hashes; uint ms; }; class Client; enum ClientWorkState { Active = 0, Deleting, Deleted }; class VersionChecker { public: VersionChecker(std::string const& _dbPath); void setOk(); bool ok() const { return m_ok; } private: bool m_ok; std::string m_path; }; static const int GenesisBlock = INT_MIN; struct PastMessage { PastMessage(Manifest const& _m, std::vector _path, Address _o): to(_m.to), from(_m.from), value(_m.value), input(_m.input), output(_m.output), path(_path), origin(_o) {} PastMessage& polish(h256 _b, u256 _ts, unsigned _n, Address _coinbase) { block = _b; timestamp = _ts; number = _n; coinbase = _coinbase; return *this; } Address to; ///< The receiving address of the transaction. Address() in the case of a creation. Address from; ///< The receiving address of the transaction. Address() in the case of a creation. u256 value; ///< The value associated with the call. bytes input; ///< The data associated with the message, or the initialiser if it's a creation transaction. bytes output; ///< The data returned by the message, or the body code if it's a creation transaction. std::vector path; ///< Call path into the block transaction. size() is always > 0. First item is the transaction index in the block. Address origin; ///< Originating sender of the transaction. Address coinbase; ///< Block coinbase. h256 block; ///< Block hash. u256 timestamp; ///< Block timestamp. unsigned number; ///< Block number. }; typedef std::vector PastMessages; class MessageFilter { public: MessageFilter(int _earliest = 0, int _latest = -1, unsigned _max = 10, unsigned _skip = 0): m_earliest(_earliest), m_latest(_latest), m_max(_max), m_skip(_skip) {} void fillStream(RLPStream& _s) const; h256 sha3() const; int earliest() const { return m_earliest; } int latest() const { return m_latest; } unsigned max() const { return m_max; } unsigned skip() const { return m_skip; } bool matches(h256 _bloom) const; bool matches(State const& _s, unsigned _i) const; PastMessages matches(Manifest const& _m, unsigned _i) const; MessageFilter from(Address _a) { m_from.insert(_a); return *this; } MessageFilter to(Address _a) { m_to.insert(_a); return *this; } MessageFilter altered(Address _a, u256 _l) { m_stateAltered.insert(std::make_pair(_a, _l)); return *this; } MessageFilter altered(Address _a) { m_altered.insert(_a); return *this; } MessageFilter withMax(unsigned _m) { m_max = _m; return *this; } MessageFilter withSkip(unsigned _m) { m_skip = _m; return *this; } MessageFilter withEarliest(int _e) { m_earliest = _e; return *this; } MessageFilter withLatest(int _e) { m_latest = _e; return *this; } private: bool matches(Manifest const& _m, std::vector _p, Address _o, PastMessages _limbo, PastMessages& o_ret) const; std::set
m_from; std::set
m_to; std::set> m_stateAltered; std::set
m_altered; int m_earliest = 0; int m_latest = -1; unsigned m_max; unsigned m_skip; }; struct InstalledFilter { InstalledFilter(MessageFilter const& _f): filter(_f) {} MessageFilter filter; unsigned refCount = 1; }; static const h256 PendingChangedFilter = u256(0); static const h256 ChainChangedFilter = u256(1); struct ClientWatch { ClientWatch() {} explicit ClientWatch(h256 _id): id(_id) {} h256 id; unsigned changes = 1; }; struct WatchChannel: public LogChannel { static const char* name() { return "(o)"; } static const int verbosity = 6; }; #define cwatch eth::LogOutputStream() struct WorkInChannel: public LogChannel { static const char* name() { return ">W>"; } static const int verbosity = 5; }; struct WorkOutChannel: public LogChannel { static const char* name() { return "() #define cworkin eth::LogOutputStream() #define cworkout eth::LogOutputStream() /** * @brief Main API hub for interfacing with Ethereum. */ class Client { public: /// Constructor. explicit Client(std::string const& _clientVersion, Address _us = Address(), std::string const& _dbPath = std::string(), bool _forceClean = false); /// Destructor. ~Client(); /// Submits the given message-call transaction. void transact(Secret _secret, u256 _value, Address _dest, bytes const& _data = bytes(), u256 _gas = 10000, u256 _gasPrice = 10 * szabo); /// Submits a new contract-creation transaction. /// @returns the new contract's address (assuming it all goes through). Address transact(Secret _secret, u256 _endowment, bytes const& _init, u256 _gas = 10000, u256 _gasPrice = 10 * szabo); /// Injects the RLP-encoded transaction given by the _rlp into the transaction queue directly. void inject(bytesConstRef _rlp); /// Blocks until all pending transactions have been processed. void flushTransactions(); /// Makes the given call. Nothing is recorded into the state. bytes call(Secret _secret, u256 _value, Address _dest, bytes const& _data = bytes(), u256 _gas = 10000, u256 _gasPrice = 10 * szabo); // Informational stuff // [NEW API] int getDefault() const { return m_default; } void setDefault(int _block) { m_default = _block; } u256 balanceAt(Address _a) const { return balanceAt(_a, m_default); } u256 countAt(Address _a) const { return countAt(_a, m_default); } u256 stateAt(Address _a, u256 _l) const { return stateAt(_a, _l, m_default); } bytes codeAt(Address _a) const { return codeAt(_a, m_default); } std::map storageAt(Address _a) const { return storageAt(_a, m_default); } u256 balanceAt(Address _a, int _block) const; u256 countAt(Address _a, int _block) const; u256 stateAt(Address _a, u256 _l, int _block) const; bytes codeAt(Address _a, int _block) const; std::map storageAt(Address _a, int _block) const; unsigned installWatch(MessageFilter const& _filter); unsigned installWatch(h256 _filterId); void uninstallWatch(unsigned _watchId); bool peekWatch(unsigned _watchId) const { std::lock_guard l(m_filterLock); try { return m_watches.at(_watchId).changes != 0; } catch (...) { return false; } } bool checkWatch(unsigned _watchId) { std::lock_guard l(m_filterLock); bool ret = false; try { ret = m_watches.at(_watchId).changes != 0; m_watches.at(_watchId).changes = 0; } catch (...) {} return ret; } PastMessages messages(unsigned _watchId) const { try { std::lock_guard l(m_filterLock); return messages(m_filters.at(m_watches.at(_watchId).id).filter); } catch (...) { return PastMessages(); } } PastMessages messages(MessageFilter const& _filter) const; // [EXTRA API]: /// Get a map containing each of the pending transactions. /// @TODO: Remove in favour of transactions(). Transactions pending() const { return m_postMine.pending(); } /// Differences between transactions. StateDiff diff(unsigned _txi) const { return diff(_txi, m_default); } StateDiff diff(unsigned _txi, h256 _block) const; StateDiff diff(unsigned _txi, int _block) const; /// Get a list of all active addresses. std::vector
addresses() const { return addresses(m_default); } std::vector
addresses(int _block) const; /// Get the fee associated for a transaction with the given data. static u256 txGas(uint _dataCount, u256 _gas = 0) { return c_txDataGas * _dataCount + c_txGas + _gas; } // [PRIVATE API - only relevant for base clients, not available in general] eth::State state(unsigned _txi, h256 _block) const; eth::State state(h256 _block) const; eth::State state(unsigned _txi) const; /// Get the object representing the current state of Ethereum. eth::State postState() const { ReadGuard l(x_stateDB); return m_postMine; } /// Get the object representing the current canonical blockchain. BlockChain const& blockChain() const { return m_bc; } // Misc stuff: void setClientVersion(std::string const& _name) { m_clientVersion = _name; } // Network stuff: /// Get information on the current peer set. std::vector peers(); /// Same as peers().size(), but more efficient. size_t peerCount() const; /// Same as peers().size(), but more efficient. void setIdealPeerCount(size_t _n) const; /// Start the network subsystem. void startNetwork(unsigned short _listenPort = 30303, std::string const& _remoteHost = std::string(), unsigned short _remotePort = 30303, NodeMode _mode = NodeMode::Full, unsigned _peers = 5, std::string const& _publicIP = std::string(), bool _upnp = true); /// Connect to a particular peer. void connect(std::string const& _seedHost, unsigned short _port = 30303); /// Stop the network subsystem. void stopNetwork(); /// Is the network subsystem up? bool haveNetwork() { ReadGuard l(x_net); return !!m_net; } /// Save peers bytes savePeers(); /// Restore peers void restorePeers(bytesConstRef _saved); // Mining stuff: /// Check block validity prior to mining. bool paranoia() const { return m_paranoia; } /// Change whether we check block validity prior to mining. void setParanoia(bool _p) { m_paranoia = _p; } /// Set the coinbase address. void setAddress(Address _us) { m_preMine.setAddress(_us); } /// Get the coinbase address. Address address() const { return m_preMine.address(); } /// Start mining. void startMining(); /// Stop mining. void stopMining(); /// Are we mining now? bool isMining() { return m_doMine; } /// Register a callback for information concerning mining. /// This callback will be in an arbitrary thread, blocking progress. JUST COPY THE DATA AND GET OUT. /// Check the progress of the mining. MineProgress miningProgress() const { return m_mineProgress; } /// Get and clear the mining history. std::list miningHistory() { auto ret = m_mineHistory; m_mineHistory.clear(); return ret; } bool forceMining() const { return m_forceMining; } void setForceMining(bool _enable) { m_forceMining = _enable; } /// Clears pending transactions. Just for debug use. void clearPending(); private: /// Ensure the worker thread is running. Needed for blockchain maintenance & mining. void ensureWorking(); /// Do some work. Handles blockchain maintenance and mining. /// @param _justQueue If true will only processing the transaction queues. void work(bool _justQueue = false); /// Do some work on the network. void workNet(); /// Collate the changed filters for the bloom filter of the given pending transaction. /// Insert any filters that are activated into @a o_changed. void appendFromNewPending(h256 _pendingTransactionBloom, h256Set& o_changed) const; /// Collate the changed filters for the hash of the given block. /// Insert any filters that are activated into @a o_changed. void appendFromNewBlock(h256 _blockHash, h256Set& o_changed) const; /// Record that the set of filters @a _filters have changed. /// This doesn't actually make any callbacks, but incrememnts some counters in m_watches. void noteChanged(h256Set const& _filters); /// Return the actual block number of the block with the given int-number (positive is the same, INT_MIN is genesis block, < 0 is negative age, thus -1 is most recently mined, 0 is pending. unsigned numberOf(int _b) const; State asOf(int _h) const; State asOf(unsigned _h) const; std::string m_clientVersion; ///< Our end-application client's name/version. VersionChecker m_vc; ///< Dummy object to check & update the protocol version. BlockChain m_bc; ///< Maintains block database. TransactionQueue m_tq; ///< Maintains a list of incoming transactions not yet in a block on the blockchain. BlockQueue m_bq; ///< Maintains a list of incoming blocks not yet on the blockchain (to be imported). mutable boost::shared_mutex x_stateDB; // TODO: remove in favour of copying m_stateDB as required and thread-safing/copying State. Have a good think about what state objects there should be. Probably want 4 (pre, post, mining, user-visible). OverlayDB m_stateDB; ///< Acts as the central point for the state database, so multiple States can share it. State m_preMine; ///< The present state of the client. State m_postMine; ///< The state of the client which we're mining (i.e. it'll have all the rewards added). std::unique_ptr m_workNet; ///< The network thread. std::atomic m_workNetState; mutable boost::shared_mutex x_net; ///< Lock for the network existance. std::unique_ptr m_net; ///< Should run in background and send us events when blocks found and allow us to send blocks as required. std::unique_ptr m_work; ///< The work thread. std::atomic m_workState; bool m_paranoia = false; bool m_doMine = false; ///< Are we supposed to be mining? bool m_forceMining = false; ///< Mine even when there are no transactions pending? MineProgress m_mineProgress; std::list m_mineHistory; mutable bool m_restartMining = false; mutable unsigned m_pendingCount = 0; mutable std::mutex m_filterLock; std::map m_filters; std::map m_watches; int m_default = -1; }; class Watch; } namespace std { void swap(eth::Watch& _a, eth::Watch& _b); } namespace eth { class Watch: public boost::noncopyable { friend void std::swap(Watch& _a, Watch& _b); public: Watch() {} Watch(Client& _c, h256 _f): m_c(&_c), m_id(_c.installWatch(_f)) {} Watch(Client& _c, MessageFilter const& _tf): m_c(&_c), m_id(_c.installWatch(_tf)) {} ~Watch() { if (m_c) m_c->uninstallWatch(m_id); } bool check() { return m_c ? m_c->checkWatch(m_id) : false; } bool peek() { return m_c ? m_c->peekWatch(m_id) : false; } PastMessages messages() const { return m_c->messages(m_id); } private: Client* m_c; unsigned m_id; }; } namespace std { inline void swap(eth::Watch& _a, eth::Watch& _b) { swap(_a.m_c, _b.m_c); swap(_a.m_id, _b.m_id); } }