/* 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 EthereumPeer.h * @author Gav Wood * @date 2014 */ #pragma once #include #include #include #include // Make sure boost/asio.hpp is included before windows.h. #include #include #include #include #include #include #include "CommonNet.h" #include "DownloadMan.h" namespace dev { namespace eth { /** * @brief The EthereumPeer class * @todo Document fully. * @todo make state transitions thread-safe. */ class EthereumPeer: public p2p::Capability { friend class EthereumHost; //TODO: remove this friend class BlockChainSync; //TODO: remove this friend class PV60Sync; //TODO: remove this friend class PV61Sync; //TODO: remove this public: /// Basic constructor. EthereumPeer(p2p::Session* _s, p2p::HostCapabilityFace* _h, unsigned _i, p2p::CapDesc const& _cap); /// Basic destructor. virtual ~EthereumPeer(); /// What is our name? static std::string name() { return "eth"; } /// What is our version? static u256 version() { return c_protocolVersion; } /// How many message types do we have? static unsigned messageCount() { return PacketCount; } /// What is the ethereum subprotocol host object. EthereumHost* host() const; /// Abort sync and reset fetch void setIdle(); /// Request hashes by number. v61+ protocol version only void requestHashes(u256 _number, unsigned _count); /// Request hashes for given parent hash. void requestHashes(h256 const& _lastHash); /// Request blocks. Uses block download manager. void requestBlocks(); /// Check if this node is rude. bool isRude() const; /// Set that it's a rude node. void setRude(); private: using p2p::Capability::sealAndSend; /// Figure out the amount of blocks we should be asking for. unsigned askOverride() const; /// Interpret an incoming message. virtual bool interpret(unsigned _id, RLP const& _r); /// Request status. Called from constructor void requestStatus(); /// Abort the sync operation. void abortSync(); /// Clear all known transactions. void clearKnownTransactions() { std::lock_guard l(x_knownTransactions); m_knownTransactions.clear(); } /// Update our asking state. void setAsking(Asking _g); /// Do we presently need syncing with this peer? bool needsSyncing() const { return !isRude() && !!m_latestHash; } /// Are we presently in the process of communicating with this peer? bool isConversing() const; /// Are we presently in a critical part of the syncing process with this peer? bool isCriticalSyncing() const; /// Runs period checks to check up on the peer. void tick(); /// Peer's protocol version. unsigned m_protocolVersion; /// Peer's network id. u256 m_networkId; /// What, if anything, we last asked the other peer for. Asking m_asking = Asking::Nothing; /// When we asked for it. Allows a time out. std::chrono::system_clock::time_point m_lastAsk; /// These are determined through either a Status message or from NewBlock. h256 m_latestHash; ///< Peer's latest block's hash that we know about or default null value if no need to sync. u256 m_totalDifficulty; ///< Peer's latest block's total difficulty. h256 m_genesisHash; ///< Peer's genesis hash u256 m_latestBlockNumber; ///< Number of the latest block this peer has /// This is built as we ask for hashes. Once no more hashes are given, we present this to the /// host who initialises the DownloadMan and m_sub becomes active for us to begin asking for blocks. unsigned m_expectedHashes = 0; ///< Estimated upper bound of hashes to expect from this peer. u256 m_syncHashNumber = 0; ///< Number of latest hash we sync to (PV61+) h256 m_syncHash; ///< Latest hash we sync to (PV60) /// Once we're asking for blocks, this becomes in use. DownloadSub m_sub; u256 m_peerCapabilityVersion; ///< Protocol version this peer supports received as capability /// Have we received a GetTransactions packet that we haven't yet answered? bool m_requireTransactions = false; Mutex x_knownBlocks; h256Hash m_knownBlocks; ///< Blocks that the peer already knows about (that don't need to be sent to them). Mutex x_knownTransactions; h256Hash m_knownTransactions; ///< Transactions that the peer already knows of. }; } }