mirror of https://github.com/PurpleI2P/i2pd.git
I2P: End-to-End encrypted and anonymous Internet
https://i2pd.website/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
169 lines
7.0 KiB
169 lines
7.0 KiB
/* |
|
* Copyright (c) 2013-2024, The PurpleI2P Project |
|
* |
|
* This file is part of Purple i2pd project and licensed under BSD3 |
|
* |
|
* See full license text in LICENSE file at top of project tree |
|
*/ |
|
|
|
#ifndef TUNNEL_POOL__ |
|
#define TUNNEL_POOL__ |
|
|
|
#include <inttypes.h> |
|
#include <set> |
|
#include <vector> |
|
#include <utility> |
|
#include <mutex> |
|
#include <memory> |
|
#include <random> |
|
#include "Identity.h" |
|
#include "LeaseSet.h" |
|
#include "RouterInfo.h" |
|
#include "I2NPProtocol.h" |
|
#include "TunnelBase.h" |
|
#include "RouterContext.h" |
|
#include "Garlic.h" |
|
|
|
namespace i2p |
|
{ |
|
namespace tunnel |
|
{ |
|
const int TUNNEL_POOL_MANAGE_INTERVAL = 10; // in seconds |
|
const int TUNNEL_POOL_MAX_INBOUND_TUNNELS_QUANTITY = 16; |
|
const int TUNNEL_POOL_MAX_OUTBOUND_TUNNELS_QUANTITY = 16; |
|
const int TUNNEL_POOL_MAX_NUM_BUILD_REQUESTS = 3; |
|
const int TUNNEL_POOL_MAX_HOP_SELECTION_ATTEMPTS = 3; |
|
|
|
class Tunnel; |
|
class InboundTunnel; |
|
class OutboundTunnel; |
|
|
|
typedef std::shared_ptr<const i2p::data::IdentityEx> Peer; |
|
struct Path |
|
{ |
|
std::vector<Peer> peers; |
|
bool isShort = true; |
|
i2p::data::RouterInfo::CompatibleTransports farEndTransports = i2p::data::RouterInfo::eAllTransports; |
|
|
|
void Add (std::shared_ptr<const i2p::data::RouterInfo> r); |
|
void Reverse (); |
|
}; |
|
|
|
/** interface for custom tunnel peer selection algorithm */ |
|
struct ITunnelPeerSelector |
|
{ |
|
virtual ~ITunnelPeerSelector() {}; |
|
virtual bool SelectPeers(Path & peers, int hops, bool isInbound) = 0; |
|
}; |
|
|
|
class TunnelPool: public std::enable_shared_from_this<TunnelPool> // per local destination |
|
{ |
|
typedef std::function<std::shared_ptr<const i2p::data::RouterInfo>(std::shared_ptr<const i2p::data::RouterInfo>, bool, bool)> SelectHopFunc; |
|
public: |
|
|
|
TunnelPool (int numInboundHops, int numOutboundHops, int numInboundTunnels, |
|
int numOutboundTunnels, int inboundVariance, int outboundVariance); |
|
~TunnelPool (); |
|
|
|
std::shared_ptr<i2p::garlic::GarlicDestination> GetLocalDestination () const { return m_LocalDestination; }; |
|
void SetLocalDestination (std::shared_ptr<i2p::garlic::GarlicDestination> destination) { m_LocalDestination = destination; }; |
|
void SetExplicitPeers (std::shared_ptr<std::vector<i2p::data::IdentHash> > explicitPeers); |
|
|
|
void CreateTunnels (); |
|
void TunnelCreated (std::shared_ptr<InboundTunnel> createdTunnel); |
|
void TunnelExpired (std::shared_ptr<InboundTunnel> expiredTunnel); |
|
void TunnelCreated (std::shared_ptr<OutboundTunnel> createdTunnel); |
|
void TunnelExpired (std::shared_ptr<OutboundTunnel> expiredTunnel); |
|
void RecreateInboundTunnel (std::shared_ptr<InboundTunnel> tunnel); |
|
void RecreateOutboundTunnel (std::shared_ptr<OutboundTunnel> tunnel); |
|
std::vector<std::shared_ptr<InboundTunnel> > GetInboundTunnels (int num) const; |
|
std::shared_ptr<OutboundTunnel> GetNextOutboundTunnel (std::shared_ptr<OutboundTunnel> excluded = nullptr, |
|
i2p::data::RouterInfo::CompatibleTransports compatible = i2p::data::RouterInfo::eAllTransports); |
|
std::shared_ptr<InboundTunnel> GetNextInboundTunnel (std::shared_ptr<InboundTunnel> excluded = nullptr, |
|
i2p::data::RouterInfo::CompatibleTransports compatible = i2p::data::RouterInfo::eAllTransports); |
|
std::pair<std::shared_ptr<OutboundTunnel>, bool> GetNewOutboundTunnel (std::shared_ptr<OutboundTunnel> old); |
|
void ManageTunnels (uint64_t ts); |
|
void ProcessGarlicMessage (std::shared_ptr<I2NPMessage> msg); |
|
void ProcessDeliveryStatus (std::shared_ptr<I2NPMessage> msg); |
|
void ProcessTunnelTest (std::shared_ptr<I2NPMessage> msg); |
|
bool ProcessTunnelTest (uint32_t msgID, uint64_t timestamp); |
|
|
|
bool IsExploratory () const; |
|
bool IsActive () const { return m_IsActive; }; |
|
void SetActive (bool isActive) { m_IsActive = isActive; }; |
|
void DetachTunnels (); |
|
|
|
int GetNumInboundTunnels () const { return m_NumInboundTunnels; }; |
|
int GetNumOutboundTunnels () const { return m_NumOutboundTunnels; }; |
|
int GetNumInboundHops() const { return m_NumInboundHops; }; |
|
int GetNumOutboundHops() const { return m_NumOutboundHops; }; |
|
|
|
/** i2cp reconfigure */ |
|
bool Reconfigure(int inboundHops, int outboundHops, int inboundQuant, int outboundQuant); |
|
|
|
void SetCustomPeerSelector(ITunnelPeerSelector * selector); |
|
void UnsetCustomPeerSelector(); |
|
bool HasCustomPeerSelector(); |
|
|
|
/** @brief make this tunnel pool yield tunnels that fit latency range [min, max] */ |
|
void RequireLatency(int min, int max) { m_MinLatency = min; m_MaxLatency = max; } |
|
|
|
/** @brief return true if this tunnel pool has a latency requirement */ |
|
bool HasLatencyRequirement() const { return m_MinLatency > 0 && m_MaxLatency > 0; } |
|
|
|
/** @brief get the lowest latency tunnel in this tunnel pool regardless of latency requirements */ |
|
std::shared_ptr<InboundTunnel> GetLowestLatencyInboundTunnel(std::shared_ptr<InboundTunnel> exclude = nullptr) const; |
|
std::shared_ptr<OutboundTunnel> GetLowestLatencyOutboundTunnel(std::shared_ptr<OutboundTunnel> exclude = nullptr) const; |
|
|
|
// for overriding tunnel peer selection |
|
std::shared_ptr<const i2p::data::RouterInfo> SelectNextHop (std::shared_ptr<const i2p::data::RouterInfo> prevHop, bool reverse, bool endpoint) const; |
|
bool StandardSelectPeers(Path & path, int numHops, bool inbound, SelectHopFunc nextHop); |
|
|
|
std::mt19937& GetRng () { return m_Rng; } |
|
|
|
private: |
|
|
|
void TestTunnels (); |
|
void CreateInboundTunnel (); |
|
void CreateOutboundTunnel (); |
|
void CreatePairedInboundTunnel (std::shared_ptr<OutboundTunnel> outboundTunnel); |
|
template<class TTunnels> |
|
typename TTunnels::value_type GetNextTunnel (TTunnels& tunnels, |
|
typename TTunnels::value_type excluded, i2p::data::RouterInfo::CompatibleTransports compatible); |
|
bool SelectPeers (Path& path, bool isInbound); |
|
bool SelectExplicitPeers (Path& path, bool isInbound); |
|
bool ValidatePeers (std::vector<std::shared_ptr<const i2p::data::IdentityEx> >& peers) const; |
|
|
|
private: |
|
|
|
std::shared_ptr<i2p::garlic::GarlicDestination> m_LocalDestination; |
|
int m_NumInboundHops, m_NumOutboundHops, m_NumInboundTunnels, m_NumOutboundTunnels, |
|
m_InboundVariance, m_OutboundVariance; |
|
std::shared_ptr<std::vector<i2p::data::IdentHash> > m_ExplicitPeers; |
|
mutable std::mutex m_InboundTunnelsMutex; |
|
std::set<std::shared_ptr<InboundTunnel>, TunnelCreationTimeCmp> m_InboundTunnels; // recent tunnel appears first |
|
mutable std::mutex m_OutboundTunnelsMutex; |
|
std::set<std::shared_ptr<OutboundTunnel>, TunnelCreationTimeCmp> m_OutboundTunnels; |
|
mutable std::mutex m_TestsMutex; |
|
std::map<uint32_t, std::pair<std::shared_ptr<OutboundTunnel>, std::shared_ptr<InboundTunnel> > > m_Tests; |
|
bool m_IsActive; |
|
uint64_t m_NextManageTime; // in seconds |
|
std::mutex m_CustomPeerSelectorMutex; |
|
ITunnelPeerSelector * m_CustomPeerSelector; |
|
|
|
int m_MinLatency = 0; // if > 0 this tunnel pool will try building tunnels with minimum latency by ms |
|
int m_MaxLatency = 0; // if > 0 this tunnel pool will try building tunnels with maximum latency by ms |
|
|
|
std::mt19937 m_Rng; |
|
|
|
public: |
|
|
|
// for HTTP only |
|
const decltype(m_OutboundTunnels)& GetOutboundTunnels () const { return m_OutboundTunnels; }; |
|
const decltype(m_InboundTunnels)& GetInboundTunnels () const { return m_InboundTunnels; }; |
|
|
|
}; |
|
} |
|
} |
|
|
|
#endif
|
|
|