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.
141 lines
4.8 KiB
141 lines
4.8 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.
|
|
|
|
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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
/** @file Session.h
|
|
* @author Gav Wood <i@gavwood.com>
|
|
* @author Alex Leverington <nessence@gmail.com>
|
|
* @date 2014
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <mutex>
|
|
#include <array>
|
|
#include <deque>
|
|
#include <set>
|
|
#include <memory>
|
|
#include <utility>
|
|
|
|
#include <libdevcore/Common.h>
|
|
#include <libdevcore/RLP.h>
|
|
#include <libdevcore/Guards.h>
|
|
#include "RLPXFrameCoder.h"
|
|
#include "RLPXSocket.h"
|
|
#include "Common.h"
|
|
|
|
namespace dev
|
|
{
|
|
|
|
namespace p2p
|
|
{
|
|
|
|
class Peer;
|
|
class ReputationManager;
|
|
|
|
/**
|
|
* @brief The Session class
|
|
* @todo Document fully.
|
|
*/
|
|
class Session: public std::enable_shared_from_this<Session>
|
|
{
|
|
friend class Host;
|
|
friend class HostCapabilityFace;
|
|
|
|
public:
|
|
Session(Host* _server, std::unique_ptr<RLPXFrameCoder>&& _io, std::shared_ptr<RLPXSocket> const& _s, std::shared_ptr<Peer> const& _n, PeerSessionInfo _info);
|
|
virtual ~Session();
|
|
|
|
void start();
|
|
void disconnect(DisconnectReason _reason);
|
|
|
|
void ping();
|
|
|
|
bool isConnected() const { return m_socket->ref().is_open(); }
|
|
|
|
NodeID id() const;
|
|
unsigned socketId() const { Guard l(x_info); return m_info.socketId; }
|
|
|
|
template <class PeerCap>
|
|
std::shared_ptr<PeerCap> cap() const { try { return std::static_pointer_cast<PeerCap>(m_capabilities.at(std::make_pair(PeerCap::name(), PeerCap::version()))); } catch (...) { return nullptr; } }
|
|
template <class PeerCap>
|
|
std::shared_ptr<PeerCap> cap(u256 const& _version) const { try { return std::static_pointer_cast<PeerCap>(m_capabilities.at(std::make_pair(PeerCap::name(), _version))); } catch (...) { return nullptr; } }
|
|
|
|
static RLPStream& prep(RLPStream& _s, PacketType _t, unsigned _args = 0);
|
|
void sealAndSend(RLPStream& _s);
|
|
|
|
ReputationManager& repMan() const;
|
|
int rating() const;
|
|
void addRating(int _r);
|
|
|
|
void addNote(std::string const& _k, std::string const& _v) { Guard l(x_info); m_info.notes[_k] = _v; }
|
|
|
|
PeerSessionInfo info() const { Guard l(x_info); return m_info; }
|
|
|
|
void ensureNodesRequested();
|
|
void serviceNodesRequest();
|
|
|
|
private:
|
|
void send(bytes&& _msg);
|
|
|
|
/// Drop the connection for the reason @a _r.
|
|
void drop(DisconnectReason _r);
|
|
|
|
/// Perform a read on the socket.
|
|
void doRead();
|
|
|
|
/// Check error code after reading and drop peer if error code.
|
|
bool checkRead(std::size_t _expected, boost::system::error_code _ec, std::size_t _length);
|
|
|
|
/// Perform a single round of the write operation. This could end up calling itself asynchronously.
|
|
void write();
|
|
|
|
/// Deliver RLPX packet to Session or Capability for interpretation.
|
|
bool readPacket(uint16_t _capId, PacketType _t, RLP const& _r);
|
|
|
|
/// Interpret an incoming Session packet.
|
|
bool interpret(PacketType _t, RLP const& _r);
|
|
|
|
/// @returns true iff the _msg forms a valid message for sending or receiving on the network.
|
|
static bool checkPacket(bytesConstRef _msg);
|
|
|
|
Host* m_server; ///< The host that owns us. Never null.
|
|
|
|
std::unique_ptr<RLPXFrameCoder> m_io; ///< Transport over which packets are sent.
|
|
std::shared_ptr<RLPXSocket> m_socket; ///< Socket of peer's connection.
|
|
Mutex x_writeQueue; ///< Mutex for the write queue.
|
|
std::deque<bytes> m_writeQueue; ///< The write queue.
|
|
std::array<byte, 16777216> m_data; ///< Buffer for ingress packet data.
|
|
bytes m_incoming; ///< Read buffer for ingress bytes.
|
|
|
|
unsigned m_protocolVersion = 0; ///< The protocol version of the peer.
|
|
std::shared_ptr<Peer> m_peer; ///< The Peer object.
|
|
bool m_dropped = false; ///< If true, we've already divested ourselves of this peer. We're just waiting for the reads & writes to fail before the shared_ptr goes OOS and the destructor kicks in.
|
|
|
|
mutable Mutex x_info;
|
|
PeerSessionInfo m_info; ///< Dynamic information about this peer.
|
|
|
|
bool m_theyRequestedNodes = false; ///< Has the peer requested nodes from us without receiveing an answer from us?
|
|
bool m_weRequestedNodes = false; ///< Have we requested nodes from the peer and not received an answer yet?
|
|
|
|
std::chrono::steady_clock::time_point m_connect; ///< Time point of connection.
|
|
std::chrono::steady_clock::time_point m_ping; ///< Time point of last ping.
|
|
std::chrono::steady_clock::time_point m_lastReceived; ///< Time point of last message.
|
|
|
|
std::map<CapDesc, std::shared_ptr<Capability>> m_capabilities; ///< The peer's capability set.
|
|
};
|
|
|
|
}
|
|
}
|
|
|