#include "rc_gossiper.hpp" #include #include #include #include #include namespace llarp { // 30 minutes static constexpr auto RCGossipFilterDecayInterval = 30min; // (30 minutes * 2) - 5 minutes static constexpr auto GossipOurRCInterval = (RCGossipFilterDecayInterval * 2) - (5min); RCGossiper::RCGossiper() : I_RCGossiper(), m_Filter(std::chrono::duration_cast(RCGossipFilterDecayInterval)) {} void RCGossiper::Init(ILinkManager* l, const RouterID& ourID, AbstractRouter* router) { m_OurRouterID = ourID; m_LinkManager = l; m_router = router; } bool RCGossiper::ShouldGossipOurRC(Time_t now) const { return now >= (m_LastGossipedOurRC + GossipOurRCInterval); } bool RCGossiper::IsOurRC(const RouterContact& rc) const { return rc.pubkey == m_OurRouterID; } void RCGossiper::Decay(Time_t now) { m_Filter.Decay(now); } void RCGossiper::Forget(const RouterID& pk) { m_Filter.Remove(pk); if (m_OurRouterID == pk) m_LastGossipedOurRC = 0s; } TimePoint_t RCGossiper::NextGossipAt() const { if (auto maybe = LastGossipAt()) return *maybe + GossipOurRCInterval; return DateClock_t::now(); } std::optional RCGossiper::LastGossipAt() const { if (m_LastGossipedOurRC == 0s) return std::nullopt; return DateClock_t::time_point{m_LastGossipedOurRC}; } bool RCGossiper::GossipRC(const RouterContact& rc) { // only distribute public routers if (not rc.IsPublicRouter()) return false; if (m_LinkManager == nullptr) return false; const RouterID pubkey(rc.pubkey); // filter check if (m_Filter.Contains(pubkey)) return false; m_Filter.Insert(pubkey); const auto now = time_now_ms(); // is this our rc? if (IsOurRC(rc)) { // should we gossip our rc? if (not ShouldGossipOurRC(now)) { // nah drop it return false; } // ya pop it m_LastGossipedOurRC = now; } // send a GRCM as gossip method DHTImmediateMessage gossip; gossip.msgs.emplace_back(new dht::GotRouterMessage(dht::Key_t{}, 0, {rc}, false)); std::vector gossipTo; // select peers to gossip to m_LinkManager->ForEachPeer( [&](const ILinkSession* peerSession, bool) { // ensure connected session if (not(peerSession && peerSession->IsEstablished())) return; // check if public router const auto other_rc = peerSession->GetRemoteRC(); if (not other_rc.IsPublicRouter()) return; gossipTo.emplace_back(other_rc.pubkey); }, true); std::unordered_set keys; // grab the keys we want to use std::sample( gossipTo.begin(), gossipTo.end(), std::inserter(keys, keys.end()), MaxGossipPeers, CSRNG{}); m_LinkManager->ForEachPeer([&](ILinkSession* peerSession) { if (not(peerSession && peerSession->IsEstablished())) return; // exclude from gossip as we have not selected to use it if (keys.count(peerSession->GetPubKey()) == 0) return; // encode message ILinkSession::Message_t msg{}; msg.resize(MAX_LINK_MSG_SIZE / 2); llarp_buffer_t buf(msg); if (not gossip.BEncode(&buf)) return; msg.resize(buf.cur - buf.base); m_router->NotifyRouterEvent(m_router->pubkey(), rc); // send message peerSession->SendMessageBuffer(std::move(msg), nullptr, gossip.Priority()); }); return true; } } // namespace llarp