i2pd/libi2pd/TunnelPool.h

171 lines
7.1 KiB
C
Raw Normal View History

/*
2024-02-02 00:38:12 +00:00
* 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
*/
2014-03-14 16:35:02 +00:00
#ifndef TUNNEL_POOL__
#define TUNNEL_POOL__
2014-03-17 20:50:03 +00:00
#include <inttypes.h>
#include <set>
#include <vector>
2014-03-17 20:50:03 +00:00
#include <utility>
2014-10-03 14:35:11 +00:00
#include <mutex>
2015-01-20 03:28:13 +00:00
#include <memory>
2024-02-15 02:35:41 +00:00
#include <random>
#include "Identity.h"
2014-03-14 16:35:02 +00:00
#include "LeaseSet.h"
#include "RouterInfo.h"
#include "I2NPProtocol.h"
#include "TunnelBase.h"
2014-04-02 17:14:21 +00:00
#include "RouterContext.h"
2014-10-07 00:18:18 +00:00
#include "Garlic.h"
2014-03-14 16:35:02 +00:00
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;
2024-02-02 00:38:12 +00:00
const int TUNNEL_POOL_MAX_NUM_BUILD_REQUESTS = 3;
const int TUNNEL_POOL_MAX_HOP_SELECTION_ATTEMPTS = 3;
2014-03-14 16:35:02 +00:00
class Tunnel;
class InboundTunnel;
class OutboundTunnel;
2017-04-08 19:14:47 +00:00
typedef std::shared_ptr<const i2p::data::IdentityEx> Peer;
2021-07-23 00:58:35 +00:00
struct Path
{
std::vector<Peer> peers;
bool isShort = true;
2021-11-06 14:49:18 +00:00
i2p::data::RouterInfo::CompatibleTransports farEndTransports = i2p::data::RouterInfo::eAllTransports;
2021-07-23 00:58:35 +00:00
void Add (std::shared_ptr<const i2p::data::RouterInfo> r);
void Reverse ();
};
2016-08-29 16:09:37 +00:00
/** interface for custom tunnel peer selection algorithm */
struct ITunnelPeerSelector
{
2017-04-08 19:14:47 +00:00
virtual ~ITunnelPeerSelector() {};
virtual bool SelectPeers(Path & peers, int hops, bool isInbound) = 0;
2016-08-29 16:09:37 +00:00
};
2015-01-20 03:28:13 +00:00
class TunnelPool: public std::enable_shared_from_this<TunnelPool> // per local destination
2014-03-14 16:35:02 +00:00
{
2023-07-11 17:16:35 +00:00
typedef std::function<std::shared_ptr<const i2p::data::RouterInfo>(std::shared_ptr<const i2p::data::RouterInfo>, bool, bool)> SelectHopFunc;
2014-03-14 16:35:02 +00:00
public:
TunnelPool (int numInboundHops, int numOutboundHops, int numInboundTunnels,
int numOutboundTunnels, int inboundVariance, int outboundVariance);
2014-03-14 16:35:02 +00:00
~TunnelPool ();
2021-11-01 01:20:16 +00:00
std::shared_ptr<i2p::garlic::GarlicDestination> GetLocalDestination () const { return m_LocalDestination; };
void SetLocalDestination (std::shared_ptr<i2p::garlic::GarlicDestination> destination) { m_LocalDestination = destination; };
2015-06-10 19:32:55 +00:00
void SetExplicitPeers (std::shared_ptr<std::vector<i2p::data::IdentHash> > explicitPeers);
2014-04-02 17:14:21 +00:00
void CreateTunnels ();
2015-01-27 19:55:46 +00:00
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);
2015-04-15 01:37:21 +00:00
void RecreateInboundTunnel (std::shared_ptr<InboundTunnel> tunnel);
void RecreateOutboundTunnel (std::shared_ptr<OutboundTunnel> tunnel);
2015-01-27 19:55:46 +00:00
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) const;
std::shared_ptr<InboundTunnel> GetNextInboundTunnel (std::shared_ptr<InboundTunnel> excluded = nullptr,
i2p::data::RouterInfo::CompatibleTransports compatible = i2p::data::RouterInfo::eAllTransports) const;
std::shared_ptr<OutboundTunnel> GetNewOutboundTunnel (std::shared_ptr<OutboundTunnel> old) const;
void ManageTunnels (uint64_t ts);
2015-06-16 14:14:14 +00:00
void ProcessGarlicMessage (std::shared_ptr<I2NPMessage> msg);
void ProcessDeliveryStatus (std::shared_ptr<I2NPMessage> msg);
2024-02-27 08:15:15 +00:00
void ProcessTunnelTest (std::shared_ptr<I2NPMessage> msg);
bool ProcessTunnelTest (uint32_t msgID, uint64_t timestamp);
2014-03-17 20:50:03 +00:00
bool IsExploratory () const;
bool IsActive () const { return m_IsActive; };
void SetActive (bool isActive) { m_IsActive = isActive; };
2014-10-11 13:47:24 +00:00
void DetachTunnels ();
2016-02-14 23:30:07 +00:00
int GetNumInboundTunnels () const { return m_NumInboundTunnels; };
int GetNumOutboundTunnels () const { return m_NumOutboundTunnels; };
2018-01-02 16:34:58 +00:00
int GetNumInboundHops() const { return m_NumInboundHops; };
int GetNumOutboundHops() const { return m_NumOutboundHops; };
2016-08-29 16:09:37 +00:00
2018-01-02 16:34:58 +00:00
/** i2cp reconfigure */
bool Reconfigure(int inboundHops, int outboundHops, int inboundQuant, int outboundQuant);
void SetCustomPeerSelector(ITunnelPeerSelector * selector);
void UnsetCustomPeerSelector();
bool HasCustomPeerSelector();
2016-11-15 15:20:09 +00:00
/** @brief make this tunnel pool yield tunnels that fit latency range [min, max] */
2024-02-25 20:57:57 +00:00
void RequireLatency(int min, int max) { m_MinLatency = min; m_MaxLatency = max; }
2016-11-15 15:20:09 +00:00
/** @brief return true if this tunnel pool has a latency requirement */
bool HasLatencyRequirement() const { return m_MinLatency > 0 && m_MaxLatency > 0; }
2016-11-15 15:20:09 +00:00
/** @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;
2016-11-15 20:40:09 +00:00
// for overriding tunnel peer selection
2023-07-11 17:16:35 +00:00
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);
2024-02-15 02:35:41 +00:00
std::mt19937& GetRng () { return m_Rng; }
private:
2022-01-07 18:39:12 +00:00
void TestTunnels ();
void CreateInboundTunnel ();
2014-03-16 20:03:20 +00:00
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) const;
2021-07-23 00:58:35 +00:00
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;
2015-06-10 19:32:55 +00:00
2014-03-14 16:35:02 +00:00
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;
2014-10-03 14:35:11 +00:00
mutable std::mutex m_InboundTunnelsMutex;
2015-01-27 19:55:46 +00:00
std::set<std::shared_ptr<InboundTunnel>, TunnelCreationTimeCmp> m_InboundTunnels; // recent tunnel appears first
2014-10-03 14:35:11 +00:00
mutable std::mutex m_OutboundTunnelsMutex;
2015-01-27 19:55:46 +00:00
std::set<std::shared_ptr<OutboundTunnel>, TunnelCreationTimeCmp> m_OutboundTunnels;
mutable std::mutex m_TestsMutex;
2015-01-27 19:55:46 +00:00
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;
2016-11-15 15:20:09 +00:00
2024-02-25 20:57:57 +00:00
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
2024-02-15 02:35:41 +00:00
std::random_device m_Rd;
std::mt19937 m_Rng;
2014-09-30 02:18:32 +00:00
public:
// for HTTP only
const decltype(m_OutboundTunnels)& GetOutboundTunnels () const { return m_OutboundTunnels; };
const decltype(m_InboundTunnels)& GetInboundTunnels () const { return m_InboundTunnels; };
};
2014-03-14 16:35:02 +00:00
}
}
#endif