lokinet/llarp/dht/messages/pubintro.cpp

200 lines
5.6 KiB
C++
Raw Normal View History

2018-12-15 16:21:52 +00:00
#include <dht/messages/pubintro.hpp>
2019-01-16 00:24:16 +00:00
#include <dht/context.hpp>
#include <dht/messages/gotintro.hpp>
#include <messages/dht_immediate.hpp>
#include <router/abstractrouter.hpp>
#include <routing/dht_message.hpp>
#include <nodedb.hpp>
2019-01-16 00:24:16 +00:00
namespace llarp
{
namespace dht
{
const uint64_t PublishIntroMessage::MaxPropagationDepth = 5;
PublishIntroMessage::~PublishIntroMessage() = default;
2019-01-16 00:24:16 +00:00
bool
2019-02-05 00:41:33 +00:00
PublishIntroMessage::DecodeKey(const llarp_buffer_t &key,
llarp_buffer_t *val)
2019-01-16 00:24:16 +00:00
{
bool read = false;
if(!BEncodeMaybeReadDictEntry("I", introset, read, key, val))
2019-01-16 00:24:16 +00:00
return false;
if(read)
return true;
if(!BEncodeMaybeReadDictInt("O", relayOrder, read, key, val))
return false;
if(read)
return true;
uint64_t relayedInt = (relayed ? 1 : 0);
if(!BEncodeMaybeReadDictInt("R", relayedInt, read, key, val))
2019-01-16 00:24:16 +00:00
return false;
if(read)
{
relayed = relayedInt;
return true;
}
2019-01-16 00:24:16 +00:00
if(!BEncodeMaybeReadDictInt("T", txID, read, key, val))
return false;
if(read)
return true;
2019-01-16 00:24:16 +00:00
if(!BEncodeMaybeReadDictInt("V", version, read, key, val))
return false;
if(read)
return true;
return false;
2019-01-16 00:24:16 +00:00
}
bool
PublishIntroMessage::HandleMessage(
llarp_dht_context *ctx,
std::vector< std::unique_ptr< IMessage > > &replies) const
{
2020-03-01 16:20:11 +00:00
const auto now = ctx->impl->Now();
const auto keyStr = introset.derivedSigningKey.ToHex();
auto &dht = *ctx->impl;
if(!introset.Verify(now))
2019-01-16 00:24:16 +00:00
{
llarp::LogWarn("Received PublishIntroMessage with invalid introset: ",
2020-02-20 15:36:29 +00:00
introset);
2019-01-16 00:24:16 +00:00
// don't propogate or store
replies.emplace_back(new GotIntroMessage({}, txID));
return true;
}
if(introset.IsExpired(now + llarp::service::MAX_INTROSET_TIME_DELTA))
2019-01-16 00:24:16 +00:00
{
// don't propogate or store
llarp::LogWarn("Received PublishIntroMessage with expired Introset: ",
2020-02-20 15:36:29 +00:00
introset);
2019-01-16 00:24:16 +00:00
replies.emplace_back(new GotIntroMessage({}, txID));
return true;
}
const llarp::dht::Key_t addr(introset.derivedSigningKey);
// identify closest 4 routers
static constexpr size_t StorageRedundancy = 4;
2020-02-29 00:07:35 +00:00
auto closestRCs =
dht.GetRouter()->nodedb()->FindClosestTo(addr, StorageRedundancy);
if(closestRCs.size() != StorageRedundancy)
2019-01-16 00:24:16 +00:00
{
2020-02-20 15:36:29 +00:00
llarp::LogWarn("Received PublishIntroMessage but only know ",
closestRCs.size(), " nodes");
replies.emplace_back(new GotIntroMessage({}, txID));
return true;
2019-01-16 00:24:16 +00:00
}
2020-02-20 15:36:29 +00:00
const auto &us = dht.OurKey();
// function to identify the closest 4 routers we know of for this introset
auto propagateIfNotUs = [&](size_t index) {
assert(index < StorageRedundancy);
const auto &rc = closestRCs[index];
const Key_t peer{rc.pubkey};
if(peer == us)
{
llarp::LogInfo("we are peer ", index,
" so storing instead of propagating");
dht.services()->PutNode(introset);
replies.emplace_back(new GotIntroMessage({introset}, txID));
}
else
{
llarp::LogInfo("propagating to peer ", index);
if(relayed)
{
dht.PropagateLocalIntroSet(pathID, txID, introset, peer, 0);
}
else
{
dht.PropagateIntroSetTo(From, txID, introset, peer, 0);
}
}
};
2020-02-20 15:36:29 +00:00
if(relayed)
{
if(relayOrder >= StorageRedundancy)
{
2020-02-20 15:36:29 +00:00
llarp::LogWarn(
"Received PublishIntroMessage with invalid relayOrder: ",
relayOrder);
replies.emplace_back(new GotIntroMessage({}, txID));
return true;
}
llarp::LogInfo("Relaying PublishIntroMessage for ", keyStr,
", txid=", txID);
propagateIfNotUs(relayOrder);
}
else
{
int candidateNumber = -1;
2020-02-29 00:07:35 +00:00
int index = 0;
2020-02-20 15:36:29 +00:00
for(const auto &rc : closestRCs)
{
2020-02-20 15:36:29 +00:00
if(rc.pubkey == dht.OurKey())
{
candidateNumber = index;
break;
}
++index;
}
if(candidateNumber >= 0)
{
LogInfo("Received PubIntro for ", keyStr, ", txid=", txID,
" and we are candidate ", candidateNumber);
dht.services()->PutNode(introset);
replies.emplace_back(new GotIntroMessage({introset}, txID));
}
else
{
LogWarn(
"!!! Received PubIntro with relayed==false but we aren't"
" candidate, intro derived key: ",
keyStr, ", txid=", txID, ", message from: ", From);
for(size_t i = 0; i < StorageRedundancy; ++i)
{
propagateIfNotUs(i);
}
}
}
2019-01-16 00:24:16 +00:00
return true;
}
bool
PublishIntroMessage::BEncode(llarp_buffer_t *buf) const
{
if(!bencode_start_dict(buf))
return false;
if(!BEncodeWriteDictMsgType(buf, "A", "I"))
return false;
if(!BEncodeWriteDictEntry("I", introset, buf))
2019-01-16 00:24:16 +00:00
return false;
if(!BEncodeWriteDictInt("O", relayOrder, buf))
return false;
if(!BEncodeWriteDictInt("R", relayed, buf))
2019-01-16 00:24:16 +00:00
return false;
if(!BEncodeWriteDictInt("T", txID, buf))
return false;
if(!BEncodeWriteDictInt("V", LLARP_PROTO_VERSION, buf))
return false;
return bencode_end(buf);
}
} // namespace dht
} // namespace llarp