/* 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 PeerServer.h * @author Gav Wood * @date 2014 */ #pragma once #include #include #include #include #include #include #include #include #include "PeerNetwork.h" #include "Guards.h" namespace ba = boost::asio; namespace bi = boost::asio::ip; namespace eth { class RLPStream; class TransactionQueue; class BlockQueue; /** * @brief The PeerServer class * @warning None of this is thread-safe. You have been warned. */ class PeerServer { friend class PeerSession; public: /// Start server, listening for connections on the given port. PeerServer(std::string const& _clientVersion, BlockChain const& _ch, u256 _networkId, unsigned short _port, NodeMode _m = NodeMode::Full, std::string const& _publicAddress = std::string(), bool _upnp = true); /// Start server, listening for connections on a system-assigned port. PeerServer(std::string const& _clientVersion, BlockChain const& _ch, u256 _networkId, NodeMode _m = NodeMode::Full, std::string const& _publicAddress = std::string(), bool _upnp = true); /// Start server, but don't listen. PeerServer(std::string const& _clientVersion, BlockChain const& _ch, u256 _networkId, NodeMode _m = NodeMode::Full); /// Will block on network process events. ~PeerServer(); /// Closes all peers. void disconnectPeers(); static unsigned protocolVersion(); u256 networkId() { return m_networkId; } /// Connect to a peer explicitly. void connect(std::string const& _addr, unsigned short _port = 30303) noexcept; void connect(bi::tcp::endpoint const& _ep); /// Sync with the BlockChain. It might contain one of our mined blocks, we might have new candidates from the network. bool sync(TransactionQueue&, BlockQueue& _bc); /// Conduct I/O, polling, syncing, whatever. /// Ideally all time-consuming I/O is done in a background thread or otherwise asynchronously, but you get this call every 100ms or so anyway. /// This won't touch alter the blockchain. void process() { if (isInitialised()) m_ioService.poll(); } /// @returns true iff we have the a peer of the given id. bool havePeer(Public _id) const; /// Set ideal number of peers. void setIdealPeerCount(unsigned _n) { m_idealPeerCount = _n; } /// Set the mode of operation on the network. void setMode(NodeMode _m) { m_mode = _m; } /// Get peer information. std::vector peers(bool _updatePing = false) const; /// Get number of peers connected; equivalent to, but faster than, peers().size(). size_t peerCount() const { Guard l(x_peers); return m_peers.size(); } /// Ping the peers, to update the latency information. void pingAll(); /// Get the port we're listening on currently. unsigned short listenPort() const { return m_public.port(); } /// Serialise the set of known peers. bytes savePeers() const; /// Deserialise the data and populate the set of known peers. void restorePeers(bytesConstRef _b); void registerPeer(std::shared_ptr _s); private: /// Session wants to pass us a block that we might not have. /// @returns true if we didn't have it. bool noteBlock(h256 _hash, bytesConstRef _data); /// Session has finished getting the chain of hashes. void noteHaveChain(std::shared_ptr const& _who); /// Called when the session has provided us with a new peer we can connect to. void noteNewPeers() {} void seal(bytes& _b); void populateAddresses(); void determinePublic(std::string const& _publicAddress, bool _upnp); void ensureAccepting(); void growPeers(); void prunePeers(); void maintainTransactions(TransactionQueue& _tq, h256 _currentBlock); void maintainBlocks(BlockQueue& _bq, h256 _currentBlock); /// Get a bunch of needed blocks. /// Removes them from our list of needed blocks. /// @returns empty if there's no more blocks left to fetch, otherwise the blocks to fetch. h256Set neededBlocks(); /// Check to see if the network peer-state initialisation has happened. bool isInitialised() const { return m_latestBlockSent; } /// Initialises the network peer-state, doing the stuff that needs to be once-only. @returns true if it really was first. bool ensureInitialised(TransactionQueue& _tq); std::map potentialPeers(); std::string m_clientVersion; NodeMode m_mode = NodeMode::Full; unsigned short m_listenPort; BlockChain const* m_chain = nullptr; ba::io_service m_ioService; bi::tcp::acceptor m_acceptor; bi::tcp::socket m_socket; UPnP* m_upnp = nullptr; bi::tcp::endpoint m_public; KeyPair m_key; u256 m_networkId; mutable std::mutex x_peers; mutable std::map> m_peers; // mutable because we flush zombie entries (null-weakptrs) as regular maintenance from a const method. mutable std::recursive_mutex m_incomingLock; std::vector m_incomingTransactions; std::vector m_incomingBlocks; std::map> m_incomingPeers; std::vector m_freePeers; mutable std::mutex x_blocksNeeded; u256 m_totalDifficultyOfNeeded; h256s m_blocksNeeded; /// From latest to earliest. h256Set m_blocksOnWay; h256 m_latestBlockSent; std::set m_transactionsSent; std::chrono::steady_clock::time_point m_lastPeersRequest; unsigned m_idealPeerCount = 5; std::vector m_addresses; std::vector m_peerAddresses; bool m_accepting = false; }; }