1
0
mirror of https://github.com/PurpleI2P/i2pd.git synced 2025-01-22 12:24:19 +00:00
i2pd/TunnelPool.h

135 lines
5.6 KiB
C
Raw Normal View History

2014-03-14 12:35:02 -04:00
#ifndef TUNNEL_POOL__
#define TUNNEL_POOL__
2014-03-17 16:50:03 -04:00
#include <inttypes.h>
#include <set>
#include <vector>
2014-03-17 16:50:03 -04:00
#include <utility>
2014-10-03 10:35:11 -04:00
#include <mutex>
2015-01-19 22:28:13 -05:00
#include <memory>
#include "Identity.h"
2014-03-14 12:35:02 -04:00
#include "LeaseSet.h"
#include "RouterInfo.h"
#include "I2NPProtocol.h"
#include "TunnelBase.h"
2014-04-02 13:14:21 -04:00
#include "RouterContext.h"
2014-10-06 20:18:18 -04:00
#include "Garlic.h"
2014-03-14 12:35:02 -04:00
namespace i2p
{
namespace tunnel
{
class Tunnel;
class InboundTunnel;
class OutboundTunnel;
2016-11-15 15:40:09 -05:00
enum TunnelBuildResult {
eBuildResultOkay, // tunnel was built okay
eBuildResultRejected, // tunnel build was explicitly rejected
eBuildResultTimeout // tunnel build timed out
};
2016-08-29 12:09:37 -04:00
/** interface for custom tunnel peer selection algorithm */
struct ITunnelPeerSelector
{
typedef std::shared_ptr<const i2p::data::IdentityEx> Peer;
typedef std::vector<Peer> TunnelPath;
2016-11-15 15:40:09 -05:00
2016-08-29 12:09:37 -04:00
virtual bool SelectPeers(TunnelPath & peers, int hops, bool isInbound) = 0;
2016-11-15 15:40:09 -05:00
virtual bool OnBuildResult(TunnelPath & peers, bool isInbound, TunnelBuildResult result) = 0;
2016-08-29 12:09:37 -04:00
};
typedef std::shared_ptr<ITunnelPeerSelector> TunnelPeerSelector;
2015-01-19 22:28:13 -05:00
class TunnelPool: public std::enable_shared_from_this<TunnelPool> // per local destination
2014-03-14 12:35:02 -04:00
{
public:
TunnelPool (int numInboundHops, int numOutboundHops, int numInboundTunnels, int numOutboundTunnels);
2014-03-14 12:35:02 -04:00
~TunnelPool ();
2014-12-15 19:08:46 -05: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 15:32:55 -04:00
void SetExplicitPeers (std::shared_ptr<std::vector<i2p::data::IdentHash> > explicitPeers);
2014-04-02 13:14:21 -04:00
void CreateTunnels ();
2015-01-27 14:55:46 -05: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-14 21:37:21 -04:00
void RecreateInboundTunnel (std::shared_ptr<InboundTunnel> tunnel);
void RecreateOutboundTunnel (std::shared_ptr<OutboundTunnel> tunnel);
2015-01-27 14:55:46 -05:00
std::vector<std::shared_ptr<InboundTunnel> > GetInboundTunnels (int num) const;
std::shared_ptr<OutboundTunnel> GetNextOutboundTunnel (std::shared_ptr<OutboundTunnel> excluded = nullptr) const;
std::shared_ptr<InboundTunnel> GetNextInboundTunnel (std::shared_ptr<InboundTunnel> excluded = nullptr) const;
std::shared_ptr<OutboundTunnel> GetNewOutboundTunnel (std::shared_ptr<OutboundTunnel> old) const;
2014-03-17 16:50:03 -04:00
void TestTunnels ();
2015-06-16 10:14:14 -04:00
void ProcessGarlicMessage (std::shared_ptr<I2NPMessage> msg);
void ProcessDeliveryStatus (std::shared_ptr<I2NPMessage> msg);
2014-03-17 16:50:03 -04:00
bool IsActive () const { return m_IsActive; };
void SetActive (bool isActive) { m_IsActive = isActive; };
2014-10-11 09:47:24 -04:00
void DetachTunnels ();
2016-02-14 18:30:07 -05:00
int GetNumInboundTunnels () const { return m_NumInboundTunnels; };
int GetNumOutboundTunnels () const { return m_NumOutboundTunnels; };
2016-08-29 12:09:37 -04:00
void SetCustomPeerSelector(TunnelPeerSelector selector);
void UnsetCustomPeerSelector();
bool HasCustomPeerSelector();
2016-11-15 10:20:09 -05:00
/** @brief make this tunnel pool yield tunnels that fit latency range [min, max] */
void RequireLatency(uint64_t min, uint64_t max) { m_MinLatency = min; m_MaxLatency = max; }
/** @brief return true if this tunnel pool has a latency requirement */
2016-11-15 10:48:33 -05:00
bool HasLatencyRequirement() const { return m_MinLatency > 0 && m_MaxLatency > 0; }
2016-11-15 10:20:09 -05: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 15:40:09 -05:00
void OnTunnelBuildResult(std::shared_ptr<Tunnel> tunnel, TunnelBuildResult result);
2016-11-15 10:20:09 -05:00
private:
void CreateInboundTunnel ();
2014-03-16 16:03:20 -04: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) const;
std::shared_ptr<const i2p::data::RouterInfo> SelectNextHop (std::shared_ptr<const i2p::data::RouterInfo> prevHop) const;
2015-11-03 09:15:49 -05:00
bool SelectPeers (std::vector<std::shared_ptr<const i2p::data::IdentityEx> >& hops, bool isInbound);
bool SelectExplicitPeers (std::vector<std::shared_ptr<const i2p::data::IdentityEx> >& hops, bool isInbound);
2015-06-10 15:32:55 -04:00
2014-03-14 12:35:02 -04:00
private:
std::shared_ptr<i2p::garlic::GarlicDestination> m_LocalDestination;
2015-05-05 12:32:13 -04:00
int m_NumInboundHops, m_NumOutboundHops, m_NumInboundTunnels, m_NumOutboundTunnels;
2015-06-10 15:32:55 -04:00
std::shared_ptr<std::vector<i2p::data::IdentHash> > m_ExplicitPeers;
2014-10-03 10:35:11 -04:00
mutable std::mutex m_InboundTunnelsMutex;
2015-01-27 14:55:46 -05:00
std::set<std::shared_ptr<InboundTunnel>, TunnelCreationTimeCmp> m_InboundTunnels; // recent tunnel appears first
2014-10-03 10:35:11 -04:00
mutable std::mutex m_OutboundTunnelsMutex;
2015-01-27 14:55:46 -05:00
std::set<std::shared_ptr<OutboundTunnel>, TunnelCreationTimeCmp> m_OutboundTunnels;
mutable std::mutex m_TestsMutex;
2015-01-27 14:55:46 -05:00
std::map<uint32_t, std::pair<std::shared_ptr<OutboundTunnel>, std::shared_ptr<InboundTunnel> > > m_Tests;
bool m_IsActive;
std::mutex m_CustomPeerSelectorMutex;
2016-08-29 12:09:37 -04:00
TunnelPeerSelector m_CustomPeerSelector;
2016-11-15 10:20:09 -05:00
uint64_t m_MinLatency=0; // if > 0 this tunnel pool will try building tunnels with minimum latency by ms
uint64_t m_MaxLatency=0; // if > 0 this tunnel pool will try building tunnels with maximum latency by ms
2014-09-29 22: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; };
2014-03-14 12:35:02 -04:00
};
}
}
#endif