/* 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 WhisperPeer.cpp * @author Gav Wood * @date 2014 */ #include #include #include "WhisperHost.h" using namespace std; using namespace dev; using namespace dev::p2p; using namespace dev::shh; WhisperPeer::WhisperPeer(std::shared_ptr _s, HostCapabilityFace* _h, unsigned _i, CapDesc const&): Capability(_s, _h, _i) { RLPStream s; sealAndSend(prep(s, StatusPacket, 1) << version()); noteAdvertiseTopicsOfInterest(); } WhisperPeer::~WhisperPeer() { } WhisperHost* WhisperPeer::host() const { return static_cast(Capability::hostCapability()); } bool WhisperPeer::interpret(unsigned _id, RLP const& _r) { switch (_id) { case StatusPacket: { auto protocolVersion = _r[0].toInt(); clog(NetMessageSummary) << "Status: " << protocolVersion; if (protocolVersion != version()) disable("Invalid protocol version."); for (auto const& m: host()->all()) m_unseen.insert(make_pair(0, m.first)); if (session()->id() < host()->host()->id()) sendMessages(); noteAdvertiseTopicsOfInterest(); break; } case MessagesPacket: { for (auto i: _r) host()->inject(Envelope(i), this); break; } case TopicFilterPacket: { setBloom((FixedHash)_r[0]); break; } default: return false; } return true; } void WhisperPeer::sendMessages() { if (m_advertiseTopicsOfInterest) sendTopicsOfInterest(host()->bloom()); RLPStream amalg; unsigned msgCount = 0; { Guard l(x_unseen); msgCount = m_unseen.size(); while (m_unseen.size()) { auto p = *m_unseen.begin(); m_unseen.erase(m_unseen.begin()); host()->streamMessage(p.second, amalg); } } if (msgCount) { RLPStream s; prep(s, MessagesPacket, msgCount).appendRaw(amalg.out(), msgCount); sealAndSend(s); } } void WhisperPeer::noteNewMessage(h256 _h, Envelope const& _m) { Guard l(x_unseen); m_unseen.insert(make_pair(rating(_m), _h)); } void WhisperPeer::sendTopicsOfInterest(FixedHash const& _bloom) { DEV_GUARDED(x_advertiseTopicsOfInterest) m_advertiseTopicsOfInterest = false; RLPStream s; prep(s, TopicFilterPacket, 1); s << _bloom; sealAndSend(s); }