You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

177 lines
4.3 KiB

/*
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.
Foobar 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 Foobar. If not, see <http://www.gnu.org/licenses/>.
*/
/** @file PeerNetwork.h
* @author Gav Wood <i@gavwood.com>
* @date 2014
*/
#pragma once
#include <memory>
#include <utility>
#include <boost/asio.hpp>
#include <boost/asio/ip/tcp.hpp>
#include <thread>
#include "RLP.h"
#include "Common.h"
namespace ba = boost::asio;
namespace bi = boost::asio::ip;
namespace eth
{
class BlockChain;
class TransactionQueue;
enum PacketType
{
Hello = 0,
Disconnect,
Ping,
Pong,
GetPeers = 0x10,
Peers,
Transactions,
Blocks,
11 years ago
GetChain,
NotInChain
};
class PeerServer;
struct PeerInfo
{
std::string clientVersion;
std::string host;
short port;
std::chrono::steady_clock::duration lastPing;
};
class PeerSession: public std::enable_shared_from_this<PeerSession>
{
friend class PeerServer;
public:
PeerSession(PeerServer* _server, bi::tcp::socket _socket, uint _rNId);
~PeerSession();
void start();
void disconnect();
11 years ago
void ping();
bi::tcp::endpoint endpoint() const;
private:
void dropped();
void doRead();
void doWrite(std::size_t length);
11 years ago
bool interpret(RLP const& _r);
static RLPStream& prep(RLPStream& _s);
static void seal(bytes& _b);
void sealAndSend(RLPStream& _s);
void sendDestroy(bytes& _msg);
void send(bytesConstRef _msg);
PeerServer* m_server;
bi::tcp::socket m_socket;
11 years ago
std::array<byte, 65536> m_data;
PeerInfo m_info;
bytes m_incoming;
uint m_protocolVersion;
uint m_networkId;
uint m_reqNetworkId;
short m_listenPort; ///< Port that the remote client is listening on for connections. Useful for giving to peers.
11 years ago
std::chrono::steady_clock::time_point m_ping;
unsigned m_rating;
std::set<h256> m_knownBlocks;
std::set<h256> m_knownTransactions;
};
class PeerServer
{
friend class PeerSession;
public:
11 years ago
/// Start server, listening for connections on the given port.
PeerServer(std::string const& _clientVersion, BlockChain const& _ch, uint _networkId, short _port);
11 years ago
/// Start server, but don't listen.
PeerServer(std::string const& _clientVersion, uint _networkId);
~PeerServer();
11 years ago
/// Connect to a peer explicitly.
bool connect(std::string const& _addr = "127.0.0.1", uint _port = 30303);
bool connect(bi::tcp::endpoint _ep);
/// Sync with the BlockChain. It might contain one of our mined blocks, we might have new candidates from the network.
/// 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.
bool process(BlockChain& _bc, TransactionQueue&, Overlay& _o);
bool process(BlockChain& _bc);
/// Set ideal number of peers.
void setIdealPeerCount(uint _n) { m_idealPeerCount = _n; }
/// Get peer information.
std::vector<PeerInfo> peers() const;
/// Get number of peers connected; equivalent to, but faster than, peers().size().
unsigned peerCount() const { return m_peers.size(); }
/// Ping the peers, to update the latency information.
11 years ago
void pingAll();
/// Get the port we're listening on currently.
short listenPort() const { return m_acceptor.local_endpoint().port(); }
private:
void populateAddresses();
void doAccept();
std::vector<bi::tcp::endpoint> potentialPeers();
std::string m_clientVersion;
BlockChain const* m_chain = nullptr;
ba::io_service m_ioService;
bi::tcp::acceptor m_acceptor;
bi::tcp::socket m_socket;
uint m_requiredNetworkId;
std::vector<std::weak_ptr<PeerSession>> m_peers;
std::vector<bytes> m_incomingTransactions;
std::vector<bytes> m_incomingBlocks;
std::vector<bi::tcp::endpoint> m_incomingPeers;
h256 m_latestBlockSent;
std::set<h256> m_transactionsSent;
std::chrono::steady_clock::time_point m_lastPeersRequest;
unsigned m_idealPeerCount = 5;
std::vector<bi::address_v4> m_addresses;
std::vector<bi::address_v4> m_peerAddresses;
};
}