lokinet/llarp/router/rc_lookup_handler.hpp
2023-09-14 05:08:10 -07:00

178 lines
4.6 KiB
C++

#pragma once
#include <llarp/router_id.hpp>
#include <llarp/util/thread/threading.hpp>
#include <chrono>
#include <unordered_map>
#include <set>
#include <unordered_set>
#include <list>
struct llarp_dht_context;
namespace llarp
{
class NodeDB;
class EventLoop;
namespace dht
{
struct AbstractDHTMessageHandler;
} // namespace dht
namespace service
{
struct Context;
} // namespace service
struct LinkManager;
struct RouterContact;
enum class RCRequestResult
{
Success,
InvalidRouter,
RouterNotFound,
BadRC
};
using RCRequestCallback =
std::function<void(const RouterID&, const RouterContact* const, const RCRequestResult)>;
struct RCLookupHandler
{
public:
using Work_t = std::function<void(void)>;
using WorkerFunc_t = std::function<void(Work_t)>;
using CallbacksQueue = std::list<RCRequestCallback>;
~RCLookupHandler() = default;
void
AddValidRouter(const RouterID& router) EXCLUDES(_mutex);
void
RemoveValidRouter(const RouterID& router) EXCLUDES(_mutex);
void
SetRouterWhitelist(
const std::vector<RouterID>& whitelist,
const std::vector<RouterID>& greylist,
const std::vector<RouterID>& greenlist
) EXCLUDES(_mutex);
bool
HaveReceivedWhitelist() const;
void
GetRC(const RouterID& router, RCRequestCallback callback, bool forceLookup = false)
EXCLUDES(_mutex);
bool
PathIsAllowed(const RouterID& remote) const EXCLUDES(_mutex);
bool
SessionIsAllowed(const RouterID& remote) const EXCLUDES(_mutex);
bool
IsGreylisted(const RouterID& remote) const EXCLUDES(_mutex);
// "greenlist" = new routers (i.e. "green") that aren't fully funded yet
bool
IsGreenlisted(const RouterID& remote) const EXCLUDES(_mutex);
// registered just means that there is at least an operator stake, but doesn't require the node
// be fully funded, active, or not decommed. (In other words: it is any of the white, grey, or
// green list).
bool
IsRegistered(const RouterID& remote) const EXCLUDES(_mutex);
bool
CheckRC(const RouterContact& rc) const;
bool
GetRandomWhitelistRouter(RouterID& router) const EXCLUDES(_mutex);
bool
CheckRenegotiateValid(RouterContact newrc, RouterContact oldrc);
void
PeriodicUpdate(llarp_time_t now);
void
ExploreNetwork();
size_t
NumberOfStrictConnectRouters() const;
void
Init(
std::shared_ptr<dht::AbstractDHTMessageHandler> dht,
std::shared_ptr<NodeDB> nodedb,
std::shared_ptr<EventLoop> loop,
WorkerFunc_t dowork,
LinkManager* linkManager,
service::Context* hiddenServiceContext,
const std::unordered_set<RouterID>& strictConnectPubkeys,
const std::set<RouterContact>& bootstrapRCList,
bool useWhitelist_arg,
bool isServiceNode_arg);
std::unordered_set<RouterID>
Whitelist() const
{
util::Lock lock{_mutex};
return whitelistRouters;
}
private:
void
HandleDHTLookupResult(RouterID remote, const std::vector<RouterContact>& results);
bool
HavePendingLookup(RouterID remote) const EXCLUDES(_mutex);
bool
RemoteInBootstrap(const RouterID& remote) const;
void
FinalizeRequest(const RouterID& router, const RouterContact* const rc, RCRequestResult result)
EXCLUDES(_mutex);
mutable util::Mutex _mutex; // protects pendingCallbacks, whitelistRouters
std::shared_ptr<dht::AbstractDHTMessageHandler> _dht = nullptr;
std::shared_ptr<NodeDB> _nodedb;
std::shared_ptr<EventLoop> _loop;
WorkerFunc_t _work = nullptr;
service::Context* _hiddenServiceContext = nullptr;
LinkManager* _linkManager = nullptr;
/// explicit whitelist of routers we will connect to directly (not for
/// service nodes)
std::unordered_set<RouterID> _strictConnectPubkeys;
std::set<RouterContact> _bootstrapRCList;
std::unordered_set<RouterID> _bootstrapRouterIDList;
std::unordered_map<RouterID, CallbacksQueue> pendingCallbacks GUARDED_BY(_mutex);
bool useWhitelist = false;
bool isServiceNode = false;
// whitelist = active routers
std::unordered_set<RouterID> whitelistRouters GUARDED_BY(_mutex);
// greylist = fully funded, but decommissioned routers
std::unordered_set<RouterID> greylistRouters GUARDED_BY(_mutex);
// greenlist = registered but not fully-staked routers
std::unordered_set<RouterID> greenlistRouters GUARDED_BY(_mutex);
using TimePoint = std::chrono::steady_clock::time_point;
std::unordered_map<RouterID, TimePoint> _routerLookupTimes;
};
} // namespace llarp