2020-05-22 16:18:41 +03:00
|
|
|
/*
|
2024-02-02 03:38:12 +03:00
|
|
|
* Copyright (c) 2013-2024, The PurpleI2P Project
|
2020-05-22 16:18:41 +03:00
|
|
|
*
|
|
|
|
* 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 20:35:02 +04:00
|
|
|
#ifndef TUNNEL_POOL__
|
|
|
|
#define TUNNEL_POOL__
|
|
|
|
|
2014-03-18 00:50:03 +04:00
|
|
|
#include <inttypes.h>
|
2014-03-14 23:13:34 +04:00
|
|
|
#include <set>
|
|
|
|
#include <vector>
|
2014-03-18 00:50:03 +04:00
|
|
|
#include <utility>
|
2014-10-03 18:35:11 +04:00
|
|
|
#include <mutex>
|
2015-01-20 06:28:13 +03:00
|
|
|
#include <memory>
|
2024-02-15 05:35:41 +03:00
|
|
|
#include <random>
|
2014-03-14 23:13:34 +04:00
|
|
|
#include "Identity.h"
|
2014-03-14 20:35:02 +04:00
|
|
|
#include "LeaseSet.h"
|
2014-09-26 05:08:20 +04:00
|
|
|
#include "RouterInfo.h"
|
2014-03-14 23:13:34 +04:00
|
|
|
#include "I2NPProtocol.h"
|
|
|
|
#include "TunnelBase.h"
|
2014-04-02 21:14:21 +04:00
|
|
|
#include "RouterContext.h"
|
2014-10-07 04:18:18 +04:00
|
|
|
#include "Garlic.h"
|
2014-03-14 20:35:02 +04:00
|
|
|
|
|
|
|
namespace i2p
|
|
|
|
{
|
|
|
|
namespace tunnel
|
|
|
|
{
|
2020-11-16 03:38:34 +03:00
|
|
|
const int TUNNEL_POOL_MANAGE_INTERVAL = 10; // in seconds
|
2021-02-12 02:32:03 +03:00
|
|
|
const int TUNNEL_POOL_MAX_INBOUND_TUNNELS_QUANTITY = 16;
|
|
|
|
const int TUNNEL_POOL_MAX_OUTBOUND_TUNNELS_QUANTITY = 16;
|
2024-02-02 03:38:12 +03:00
|
|
|
const int TUNNEL_POOL_MAX_NUM_BUILD_REQUESTS = 3;
|
2024-02-20 18:39:48 +03:00
|
|
|
const int TUNNEL_POOL_MAX_HOP_SELECTION_ATTEMPTS = 3;
|
2021-11-27 23:30:35 +03:00
|
|
|
|
2014-03-14 20:35:02 +04:00
|
|
|
class Tunnel;
|
|
|
|
class InboundTunnel;
|
|
|
|
class OutboundTunnel;
|
|
|
|
|
2017-04-08 22:14:47 +03:00
|
|
|
typedef std::shared_ptr<const i2p::data::IdentityEx> Peer;
|
2021-07-23 03:58:35 +03:00
|
|
|
struct Path
|
|
|
|
{
|
|
|
|
std::vector<Peer> peers;
|
|
|
|
bool isShort = true;
|
2021-11-06 17:49:18 +03:00
|
|
|
i2p::data::RouterInfo::CompatibleTransports farEndTransports = i2p::data::RouterInfo::eAllTransports;
|
2021-11-27 23:30:35 +03:00
|
|
|
|
2021-07-23 03:58:35 +03:00
|
|
|
void Add (std::shared_ptr<const i2p::data::RouterInfo> r);
|
|
|
|
void Reverse ();
|
|
|
|
};
|
2017-04-08 19:51:35 +03:00
|
|
|
|
2016-08-29 19:09:37 +03:00
|
|
|
/** interface for custom tunnel peer selection algorithm */
|
|
|
|
struct ITunnelPeerSelector
|
|
|
|
{
|
2017-04-08 22:14:47 +03:00
|
|
|
virtual ~ITunnelPeerSelector() {};
|
2017-04-08 19:51:35 +03:00
|
|
|
virtual bool SelectPeers(Path & peers, int hops, bool isInbound) = 0;
|
2016-08-29 19:09:37 +03:00
|
|
|
};
|
|
|
|
|
2015-01-20 06:28:13 +03:00
|
|
|
class TunnelPool: public std::enable_shared_from_this<TunnelPool> // per local destination
|
2014-03-14 20:35:02 +04:00
|
|
|
{
|
2023-07-11 20:16:35 +03:00
|
|
|
typedef std::function<std::shared_ptr<const i2p::data::RouterInfo>(std::shared_ptr<const i2p::data::RouterInfo>, bool, bool)> SelectHopFunc;
|
2014-03-14 20:35:02 +04:00
|
|
|
public:
|
|
|
|
|
2022-05-20 19:56:05 +03:00
|
|
|
TunnelPool (int numInboundHops, int numOutboundHops, int numInboundTunnels,
|
2024-09-20 04:16:16 +03:00
|
|
|
int numOutboundTunnels, int inboundVariance, int outboundVariance, bool isHighBandwidth);
|
2014-03-14 20:35:02 +04:00
|
|
|
~TunnelPool ();
|
2017-04-08 19:51:35 +03:00
|
|
|
|
2021-11-01 04:20:16 +03: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 22:32:55 +03:00
|
|
|
void SetExplicitPeers (std::shared_ptr<std::vector<i2p::data::IdentHash> > explicitPeers);
|
2014-04-02 21:14:21 +04:00
|
|
|
|
2014-03-14 23:13:34 +04:00
|
|
|
void CreateTunnels ();
|
2015-01-27 22:55:46 +03: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 04:37:21 +03:00
|
|
|
void RecreateInboundTunnel (std::shared_ptr<InboundTunnel> tunnel);
|
|
|
|
void RecreateOutboundTunnel (std::shared_ptr<OutboundTunnel> tunnel);
|
2015-01-27 22:55:46 +03:00
|
|
|
std::vector<std::shared_ptr<InboundTunnel> > GetInboundTunnels (int num) const;
|
2021-11-05 21:51:24 +03:00
|
|
|
std::shared_ptr<OutboundTunnel> GetNextOutboundTunnel (std::shared_ptr<OutboundTunnel> excluded = nullptr,
|
2024-05-07 20:01:27 +03:00
|
|
|
i2p::data::RouterInfo::CompatibleTransports compatible = i2p::data::RouterInfo::eAllTransports);
|
2021-11-05 21:51:24 +03:00
|
|
|
std::shared_ptr<InboundTunnel> GetNextInboundTunnel (std::shared_ptr<InboundTunnel> excluded = nullptr,
|
2024-05-07 20:01:27 +03:00
|
|
|
i2p::data::RouterInfo::CompatibleTransports compatible = i2p::data::RouterInfo::eAllTransports);
|
|
|
|
std::pair<std::shared_ptr<OutboundTunnel>, bool> GetNewOutboundTunnel (std::shared_ptr<OutboundTunnel> old);
|
2020-11-16 03:38:34 +03:00
|
|
|
void ManageTunnels (uint64_t ts);
|
2015-06-16 17:14:14 +03:00
|
|
|
void ProcessGarlicMessage (std::shared_ptr<I2NPMessage> msg);
|
|
|
|
void ProcessDeliveryStatus (std::shared_ptr<I2NPMessage> msg);
|
2024-02-27 11:15:15 +03:00
|
|
|
void ProcessTunnelTest (std::shared_ptr<I2NPMessage> msg);
|
2024-02-27 19:47:32 +03:00
|
|
|
bool ProcessTunnelTest (uint32_t msgID, uint64_t timestamp);
|
2014-03-18 00:50:03 +04:00
|
|
|
|
2021-05-28 19:11:24 +03:00
|
|
|
bool IsExploratory () const;
|
2014-10-13 19:21:57 +04:00
|
|
|
bool IsActive () const { return m_IsActive; };
|
|
|
|
void SetActive (bool isActive) { m_IsActive = isActive; };
|
2014-10-11 17:47:24 +04:00
|
|
|
void DetachTunnels ();
|
2016-02-15 02:30:07 +03:00
|
|
|
|
|
|
|
int GetNumInboundTunnels () const { return m_NumInboundTunnels; };
|
|
|
|
int GetNumOutboundTunnels () const { return m_NumOutboundTunnels; };
|
2018-01-02 19:34:58 +03:00
|
|
|
int GetNumInboundHops() const { return m_NumInboundHops; };
|
|
|
|
int GetNumOutboundHops() const { return m_NumOutboundHops; };
|
2016-08-29 19:09:37 +03:00
|
|
|
|
2018-01-02 19:34:58 +03:00
|
|
|
/** i2cp reconfigure */
|
|
|
|
bool Reconfigure(int inboundHops, int outboundHops, int inboundQuant, int outboundQuant);
|
2020-03-01 13:25:50 +03:00
|
|
|
|
2017-04-08 19:51:35 +03:00
|
|
|
void SetCustomPeerSelector(ITunnelPeerSelector * selector);
|
2016-08-30 22:11:39 +03:00
|
|
|
void UnsetCustomPeerSelector();
|
|
|
|
bool HasCustomPeerSelector();
|
2016-11-15 18:20:09 +03:00
|
|
|
|
2020-03-01 13:25:50 +03:00
|
|
|
/** @brief make this tunnel pool yield tunnels that fit latency range [min, max] */
|
2024-02-25 23:57:57 +03:00
|
|
|
void RequireLatency(int min, int max) { m_MinLatency = min; m_MaxLatency = max; }
|
2016-11-15 18:20:09 +03:00
|
|
|
|
2020-03-01 13:25:50 +03: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 18:20:09 +03:00
|
|
|
|
2020-03-01 13:25:50 +03: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 23:40:09 +03:00
|
|
|
|
2020-03-01 13:25:50 +03:00
|
|
|
// for overriding tunnel peer selection
|
2023-07-11 20:16:35 +03:00
|
|
|
std::shared_ptr<const i2p::data::RouterInfo> SelectNextHop (std::shared_ptr<const i2p::data::RouterInfo> prevHop, bool reverse, bool endpoint) const;
|
2023-02-03 23:59:56 +03:00
|
|
|
bool StandardSelectPeers(Path & path, int numHops, bool inbound, SelectHopFunc nextHop);
|
2023-02-11 09:41:51 +03:00
|
|
|
|
2024-02-15 05:35:41 +03:00
|
|
|
std::mt19937& GetRng () { return m_Rng; }
|
|
|
|
|
2014-03-14 23:13:34 +04:00
|
|
|
private:
|
2017-04-08 19:51:35 +03:00
|
|
|
|
2022-01-07 21:39:12 +03:00
|
|
|
void TestTunnels ();
|
2017-04-08 19:51:35 +03:00
|
|
|
void CreateInboundTunnel ();
|
2014-03-17 00:03:20 +04:00
|
|
|
void CreateOutboundTunnel ();
|
2015-05-07 23:03:12 +03:00
|
|
|
void CreatePairedInboundTunnel (std::shared_ptr<OutboundTunnel> outboundTunnel);
|
2014-04-04 00:27:37 +04:00
|
|
|
template<class TTunnels>
|
2021-11-27 23:30:35 +03:00
|
|
|
typename TTunnels::value_type GetNextTunnel (TTunnels& tunnels,
|
2024-05-07 20:01:27 +03:00
|
|
|
typename TTunnels::value_type excluded, i2p::data::RouterInfo::CompatibleTransports compatible);
|
2021-07-23 03:58:35 +03:00
|
|
|
bool SelectPeers (Path& path, bool isInbound);
|
|
|
|
bool SelectExplicitPeers (Path& path, bool isInbound);
|
2023-04-11 06:04:38 +03:00
|
|
|
bool ValidatePeers (std::vector<std::shared_ptr<const i2p::data::IdentityEx> >& peers) const;
|
2015-06-10 22:32:55 +03:00
|
|
|
|
2014-03-14 20:35:02 +04:00
|
|
|
private:
|
|
|
|
|
2015-12-16 22:52:48 +03:00
|
|
|
std::shared_ptr<i2p::garlic::GarlicDestination> m_LocalDestination;
|
2022-05-20 19:56:05 +03:00
|
|
|
int m_NumInboundHops, m_NumOutboundHops, m_NumInboundTunnels, m_NumOutboundTunnels,
|
2022-03-08 06:20:11 +03:00
|
|
|
m_InboundVariance, m_OutboundVariance;
|
2017-04-08 19:51:35 +03:00
|
|
|
std::shared_ptr<std::vector<i2p::data::IdentHash> > m_ExplicitPeers;
|
2014-10-03 18:35:11 +04:00
|
|
|
mutable std::mutex m_InboundTunnelsMutex;
|
2015-01-27 22:55:46 +03:00
|
|
|
std::set<std::shared_ptr<InboundTunnel>, TunnelCreationTimeCmp> m_InboundTunnels; // recent tunnel appears first
|
2014-10-03 18:35:11 +04:00
|
|
|
mutable std::mutex m_OutboundTunnelsMutex;
|
2015-01-27 22:55:46 +03:00
|
|
|
std::set<std::shared_ptr<OutboundTunnel>, TunnelCreationTimeCmp> m_OutboundTunnels;
|
2016-02-17 00:10:22 +03:00
|
|
|
mutable std::mutex m_TestsMutex;
|
2015-01-27 22:55:46 +03:00
|
|
|
std::map<uint32_t, std::pair<std::shared_ptr<OutboundTunnel>, std::shared_ptr<InboundTunnel> > > m_Tests;
|
2024-09-20 04:16:16 +03:00
|
|
|
bool m_IsActive, m_IsHighBandwidth;
|
2020-11-16 03:38:34 +03:00
|
|
|
uint64_t m_NextManageTime; // in seconds
|
2016-08-30 22:11:39 +03:00
|
|
|
std::mutex m_CustomPeerSelectorMutex;
|
2017-04-08 19:51:35 +03:00
|
|
|
ITunnelPeerSelector * m_CustomPeerSelector;
|
2016-11-15 18:20:09 +03:00
|
|
|
|
2024-02-25 23:57:57 +03: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
|
2017-04-08 19:51:35 +03:00
|
|
|
|
2024-02-15 05:35:41 +03:00
|
|
|
std::mt19937 m_Rng;
|
|
|
|
|
2014-09-30 06:18:32 +04:00
|
|
|
public:
|
|
|
|
|
|
|
|
// for HTTP only
|
|
|
|
const decltype(m_OutboundTunnels)& GetOutboundTunnels () const { return m_OutboundTunnels; };
|
|
|
|
const decltype(m_InboundTunnels)& GetInboundTunnels () const { return m_InboundTunnels; };
|
|
|
|
|
2017-04-08 19:51:35 +03:00
|
|
|
};
|
2014-03-14 20:35:02 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|