lokinet/llarp/service/sendcontext.cpp

112 lines
2.9 KiB
C++
Raw Normal View History

2019-04-19 16:02:32 +00:00
#include <service/sendcontext.hpp>
#include <router/abstractrouter.hpp>
2019-06-19 22:30:07 +00:00
#include <routing/path_transfer_message.hpp>
2019-06-15 14:55:14 +00:00
#include <service/endpoint.hpp>
#include <util/logic.hpp>
2019-07-30 23:42:13 +00:00
#include <utility>
2019-04-19 16:02:32 +00:00
namespace llarp
{
namespace service
{
2019-07-30 23:42:13 +00:00
SendContext::SendContext(ServiceInfo ident, const Introduction& intro,
path::PathSet* send, Endpoint* ep)
: remoteIdent(std::move(ident))
2019-04-19 16:02:32 +00:00
, remoteIntro(intro)
, m_PathSet(send)
, m_DataHandler(ep)
, m_Endpoint(ep)
{
createdAt = ep->Now();
currentConvoTag.Zero();
}
bool
SendContext::Send(const ProtocolFrame& msg, path::Path_ptr path)
2019-04-19 16:02:32 +00:00
{
2019-07-18 16:28:17 +00:00
util::Lock lock(&m_SendQueueMutex);
m_SendQueue.emplace_back(
std::make_shared< const routing::PathTransferMessage >(
msg, remoteIntro.pathID),
path);
2019-04-25 17:15:56 +00:00
return true;
}
void
SendContext::FlushUpstream()
2019-04-25 17:15:56 +00:00
{
auto r = m_Endpoint->Router();
util::Lock lock(&m_SendQueueMutex);
for(const auto& item : m_SendQueue)
{
if(item.second->SendRoutingMessage(*item.first, r))
2019-04-19 16:02:32 +00:00
{
2019-04-25 17:15:56 +00:00
lastGoodSend = r->Now();
2019-04-19 16:02:32 +00:00
}
else
2019-07-18 16:28:17 +00:00
LogError(m_Endpoint->Name(), " failed to send frame on path");
2019-04-25 17:15:56 +00:00
}
m_SendQueue.clear();
2019-04-19 16:02:32 +00:00
}
/// send on an established convo tag
void
SendContext::EncryptAndSendTo(const llarp_buffer_t& payload, ProtocolType t)
{
SharedSecret shared;
ProtocolFrame f;
2019-04-19 16:02:32 +00:00
f.N.Randomize();
f.T = currentConvoTag;
2019-06-06 10:52:27 +00:00
f.S = ++sequenceNo;
2019-04-19 16:02:32 +00:00
auto path = m_PathSet->GetNewestPathByRouter(remoteIntro.router);
if(!path)
{
2019-07-18 16:28:17 +00:00
LogError(m_Endpoint->Name(),
" cannot encrypt and send: no path for intro ", remoteIntro);
2019-04-19 16:02:32 +00:00
return;
}
if(!m_DataHandler->GetCachedSessionKeyFor(f.T, shared))
2019-04-19 16:02:32 +00:00
{
2019-07-18 16:28:17 +00:00
LogError(m_Endpoint->Name(),
" has no cached session key on session T=", f.T);
2019-04-19 16:02:32 +00:00
return;
}
2019-05-22 16:20:50 +00:00
ProtocolMessage m;
m_DataHandler->PutIntroFor(f.T, remoteIntro);
m_DataHandler->PutReplyIntroFor(f.T, path->intro);
m.proto = t;
2019-06-06 10:52:27 +00:00
m.seqno = m_Endpoint->GetSeqNoForConvo(f.T);
m.introReply = path->intro;
f.F = m.introReply.pathID;
m.sender = m_Endpoint->GetIdentity().pub;
m.tag = f.T;
m.PutBuffer(payload);
if(!f.EncryptAndSign(m, shared, m_Endpoint->GetIdentity()))
{
2019-07-18 16:28:17 +00:00
LogError(m_Endpoint->Name(), " failed to sign message");
return;
2019-05-22 16:20:50 +00:00
}
Send(f, path);
2019-04-19 16:02:32 +00:00
}
void
SendContext::AsyncEncryptAndSendTo(const llarp_buffer_t& data,
ProtocolType protocol)
{
if(lastGoodSend)
2019-04-19 16:02:32 +00:00
{
EncryptAndSendTo(data, protocol);
}
else
{
AsyncGenIntro(data, protocol);
}
}
} // namespace service
} // namespace llarp