2020-05-22 13:18:41 +00:00
|
|
|
/*
|
2024-01-11 20:39:42 +00:00
|
|
|
* Copyright (c) 2013-2024, The PurpleI2P Project
|
2020-05-22 13:18:41 +00: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
|
|
|
|
*/
|
|
|
|
|
2013-10-27 15:26:39 +00:00
|
|
|
#ifndef TRANSPORTS_H__
|
|
|
|
#define TRANSPORTS_H__
|
|
|
|
|
|
|
|
#include <thread>
|
2014-04-04 20:29:40 +00:00
|
|
|
#include <mutex>
|
2024-11-29 02:56:26 +00:00
|
|
|
#include <future>
|
2014-04-04 20:29:40 +00:00
|
|
|
#include <condition_variable>
|
2013-10-27 15:26:39 +00:00
|
|
|
#include <functional>
|
2020-06-29 22:19:31 +00:00
|
|
|
#include <unordered_map>
|
2015-01-21 02:05:57 +00:00
|
|
|
#include <vector>
|
2014-04-04 20:29:40 +00:00
|
|
|
#include <queue>
|
2013-10-27 15:26:39 +00:00
|
|
|
#include <string>
|
2014-11-25 21:30:15 +00:00
|
|
|
#include <memory>
|
2015-03-16 23:33:59 +00:00
|
|
|
#include <atomic>
|
2013-10-27 15:26:39 +00:00
|
|
|
#include <boost/asio.hpp>
|
2014-10-20 20:09:59 +00:00
|
|
|
#include "TransportSession.h"
|
2022-03-11 21:17:44 +00:00
|
|
|
#include "SSU2.h"
|
2018-07-13 19:59:28 +00:00
|
|
|
#include "NTCP2.h"
|
2013-10-27 15:26:39 +00:00
|
|
|
#include "RouterInfo.h"
|
|
|
|
#include "I2NPProtocol.h"
|
2014-04-04 17:30:13 +00:00
|
|
|
#include "Identity.h"
|
2024-11-02 15:20:23 +00:00
|
|
|
#include "util.h"
|
2013-10-27 15:26:39 +00:00
|
|
|
|
|
|
|
namespace i2p
|
2014-10-21 16:25:53 +00:00
|
|
|
{
|
|
|
|
namespace transport
|
2013-10-27 15:26:39 +00:00
|
|
|
{
|
2020-06-30 00:02:09 +00:00
|
|
|
template<typename Keys>
|
|
|
|
class EphemeralKeysSupplier
|
2014-04-04 20:29:40 +00:00
|
|
|
{
|
2020-06-30 00:02:09 +00:00
|
|
|
// called from this file only, so implementation is in Transports.cpp
|
2014-04-04 20:29:40 +00:00
|
|
|
public:
|
|
|
|
|
2020-06-30 00:02:09 +00:00
|
|
|
EphemeralKeysSupplier (int size);
|
|
|
|
~EphemeralKeysSupplier ();
|
2014-04-04 20:29:40 +00:00
|
|
|
void Start ();
|
|
|
|
void Stop ();
|
2020-06-30 00:02:09 +00:00
|
|
|
std::shared_ptr<Keys> Acquire ();
|
|
|
|
void Return (std::shared_ptr<Keys> pair);
|
2014-04-04 20:29:40 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
void Run ();
|
2020-06-30 00:02:09 +00:00
|
|
|
void CreateEphemeralKeys (int num);
|
2014-04-04 20:29:40 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
|
2014-09-17 15:13:25 +00:00
|
|
|
const int m_QueueSize;
|
2020-06-30 00:02:09 +00:00
|
|
|
std::queue<std::shared_ptr<Keys> > m_Queue;
|
2024-11-02 15:20:23 +00:00
|
|
|
i2p::util::MemoryPoolMt<Keys> m_KeysPool;
|
2014-04-04 20:29:40 +00:00
|
|
|
|
|
|
|
bool m_IsRunning;
|
2024-11-02 15:20:23 +00:00
|
|
|
std::unique_ptr<std::thread> m_Thread;
|
2014-04-04 20:29:40 +00:00
|
|
|
std::condition_variable m_Acquired;
|
|
|
|
std::mutex m_AcquiredMutex;
|
|
|
|
};
|
2020-06-30 00:02:09 +00:00
|
|
|
typedef EphemeralKeysSupplier<i2p::crypto::X25519Keys> X25519KeysPairSupplier;
|
2021-11-27 20:30:35 +00:00
|
|
|
|
2022-10-09 17:24:43 +00:00
|
|
|
const int PEER_ROUTER_INFO_UPDATE_INTERVAL = 31*60; // in seconds
|
|
|
|
const int PEER_ROUTER_INFO_UPDATE_INTERVAL_VARIANCE = 7*60; // in seconds
|
2023-01-02 00:42:40 +00:00
|
|
|
const size_t PEER_ROUTER_INFO_OVERLOAD_QUEUE_SIZE = 25;
|
2024-08-19 19:51:56 +00:00
|
|
|
const int PEER_SELECTION_MIN_INTERVAL = 20; // in seconds
|
2015-01-13 03:53:35 +00:00
|
|
|
struct Peer
|
|
|
|
{
|
2015-01-14 02:31:39 +00:00
|
|
|
int numAttempts;
|
2015-01-13 03:53:35 +00:00
|
|
|
std::shared_ptr<const i2p::data::RouterInfo> router;
|
2015-06-09 15:00:37 +00:00
|
|
|
std::list<std::shared_ptr<TransportSession> > sessions;
|
2024-08-19 19:51:56 +00:00
|
|
|
uint64_t creationTime, nextRouterInfoUpdateTime, lastSelectionTime;
|
2024-10-29 00:36:50 +00:00
|
|
|
std::list<std::shared_ptr<i2p::I2NPMessage> > delayedMessages;
|
2022-09-30 23:24:36 +00:00
|
|
|
std::vector<i2p::data::RouterInfo::SupportedTransports> priority;
|
2024-08-18 22:34:28 +00:00
|
|
|
bool isHighBandwidth, isEligible;
|
2015-01-13 03:53:35 +00:00
|
|
|
|
2022-09-30 23:24:36 +00:00
|
|
|
Peer (std::shared_ptr<const i2p::data::RouterInfo> r, uint64_t ts):
|
|
|
|
numAttempts (0), router (r), creationTime (ts),
|
2023-02-03 20:59:56 +00:00
|
|
|
nextRouterInfoUpdateTime (ts + PEER_ROUTER_INFO_UPDATE_INTERVAL),
|
2024-08-19 19:51:56 +00:00
|
|
|
lastSelectionTime (0), isHighBandwidth (false), isEligible (false)
|
2022-09-30 23:24:36 +00:00
|
|
|
{
|
2024-08-19 18:39:07 +00:00
|
|
|
UpdateParams (router);
|
2022-10-09 17:24:43 +00:00
|
|
|
}
|
2024-01-30 00:54:43 +00:00
|
|
|
|
2015-06-09 15:00:37 +00:00
|
|
|
void Done ()
|
|
|
|
{
|
2016-08-08 22:53:37 +00:00
|
|
|
for (auto& it: sessions)
|
2015-06-09 15:00:37 +00:00
|
|
|
it->Done ();
|
2024-01-30 15:04:19 +00:00
|
|
|
// drop not sent delayed messages
|
|
|
|
for (auto& it: delayedMessages)
|
|
|
|
it->Drop ();
|
2017-05-29 05:28:16 +00:00
|
|
|
}
|
2023-02-03 20:59:56 +00:00
|
|
|
|
|
|
|
void SetRouter (std::shared_ptr<const i2p::data::RouterInfo> r)
|
|
|
|
{
|
|
|
|
router = r;
|
2024-08-19 18:39:07 +00:00
|
|
|
UpdateParams (router);
|
2023-02-11 06:41:51 +00:00
|
|
|
}
|
2024-01-19 00:18:51 +00:00
|
|
|
|
2024-08-19 18:39:07 +00:00
|
|
|
bool IsConnected () const { return !sessions.empty (); }
|
|
|
|
void UpdateParams (std::shared_ptr<const i2p::data::RouterInfo> router);
|
2017-05-29 05:28:16 +00:00
|
|
|
};
|
|
|
|
|
2022-06-19 18:21:35 +00:00
|
|
|
const uint64_t SESSION_CREATION_TIMEOUT = 15; // in seconds
|
2016-11-13 14:14:05 +00:00
|
|
|
const int PEER_TEST_INTERVAL = 71; // in minutes
|
2023-11-10 02:56:32 +00:00
|
|
|
const int PEER_TEST_DELAY_INTERVAL = 20; // in milliseconds
|
|
|
|
const int PEER_TEST_DELAY_INTERVAL_VARIANCE = 30; // in milliseconds
|
2021-05-04 18:59:25 +00:00
|
|
|
const int MAX_NUM_DELAYED_MESSAGES = 150;
|
2023-04-16 02:16:31 +00:00
|
|
|
const int CHECK_PROFILE_NUM_DELAYED_MESSAGES = 15; // check profile after
|
2024-11-01 18:46:13 +00:00
|
|
|
const int NUM_X25519_PRE_GENERATED_KEYS = 25; // pre-generated x25519 keys pairs
|
|
|
|
|
2024-02-18 20:54:43 +00:00
|
|
|
const int TRAFFIC_SAMPLE_COUNT = 301; // seconds
|
|
|
|
|
|
|
|
struct TrafficSample
|
|
|
|
{
|
|
|
|
uint64_t Timestamp;
|
|
|
|
uint64_t TotalReceivedBytes;
|
|
|
|
uint64_t TotalSentBytes;
|
|
|
|
uint64_t TotalTransitTransmittedBytes;
|
|
|
|
};
|
|
|
|
|
2013-10-27 15:26:39 +00:00
|
|
|
class Transports
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
Transports ();
|
|
|
|
~Transports ();
|
|
|
|
|
2022-11-23 00:29:20 +00:00
|
|
|
void Start (bool enableNTCP2=true, bool enableSSU2=true);
|
2013-10-27 15:26:39 +00:00
|
|
|
void Stop ();
|
2024-04-30 11:51:46 +00:00
|
|
|
bool IsRunning () const { return m_IsRunning; }
|
2016-06-13 15:34:44 +00:00
|
|
|
|
2022-07-05 06:11:16 +00:00
|
|
|
bool IsBoundSSU2() const { return m_SSU2Server != nullptr; }
|
2018-08-10 19:35:43 +00:00
|
|
|
bool IsBoundNTCP2() const { return m_NTCP2Server != nullptr; }
|
2017-05-29 05:28:16 +00:00
|
|
|
|
2016-07-13 14:09:22 +00:00
|
|
|
bool IsOnline() const { return m_IsOnline; };
|
2020-10-11 21:51:40 +00:00
|
|
|
void SetOnline (bool online);
|
2016-07-13 14:09:22 +00:00
|
|
|
|
2024-11-25 15:08:27 +00:00
|
|
|
auto& GetService () { return *m_Service; };
|
2020-06-30 00:02:09 +00:00
|
|
|
std::shared_ptr<i2p::crypto::X25519Keys> GetNextX25519KeysPair ();
|
|
|
|
void ReuseX25519KeysPair (std::shared_ptr<i2p::crypto::X25519Keys> pair);
|
2013-10-27 15:26:39 +00:00
|
|
|
|
2024-11-29 02:56:26 +00:00
|
|
|
std::future<std::shared_ptr<TransportSession> > SendMessage (const i2p::data::IdentHash& ident, std::shared_ptr<i2p::I2NPMessage> msg);
|
|
|
|
std::future<std::shared_ptr<TransportSession> > SendMessages (const i2p::data::IdentHash& ident, std::list<std::shared_ptr<i2p::I2NPMessage> >&& msgs);
|
2015-01-13 03:53:35 +00:00
|
|
|
|
|
|
|
void PeerConnected (std::shared_ptr<TransportSession> session);
|
|
|
|
void PeerDisconnected (std::shared_ptr<TransportSession> session);
|
2015-03-18 00:56:51 +00:00
|
|
|
bool IsConnected (const i2p::data::IdentHash& ident) const;
|
2017-05-29 05:28:16 +00:00
|
|
|
|
2015-03-16 23:33:59 +00:00
|
|
|
void UpdateSentBytes (uint64_t numBytes) { m_TotalSentBytes += numBytes; };
|
|
|
|
void UpdateReceivedBytes (uint64_t numBytes) { m_TotalReceivedBytes += numBytes; };
|
|
|
|
uint64_t GetTotalSentBytes () const { return m_TotalSentBytes; };
|
2017-05-02 18:20:00 +00:00
|
|
|
uint64_t GetTotalReceivedBytes () const { return m_TotalReceivedBytes; };
|
|
|
|
uint64_t GetTotalTransitTransmittedBytes () const { return m_TotalTransitTransmittedBytes; }
|
|
|
|
void UpdateTotalTransitTransmittedBytes (uint32_t add) { m_TotalTransitTransmittedBytes += add; };
|
2020-03-01 10:25:50 +00:00
|
|
|
uint32_t GetInBandwidth () const { return m_InBandwidth; };
|
2016-03-31 00:00:00 +00:00
|
|
|
uint32_t GetOutBandwidth () const { return m_OutBandwidth; };
|
2017-05-02 18:20:00 +00:00
|
|
|
uint32_t GetTransitBandwidth () const { return m_TransitBandwidth; };
|
2022-12-12 05:12:23 +00:00
|
|
|
uint32_t GetInBandwidth15s () const { return m_InBandwidth15s; };
|
|
|
|
uint32_t GetOutBandwidth15s () const { return m_OutBandwidth15s; };
|
|
|
|
uint32_t GetTransitBandwidth15s () const { return m_TransitBandwidth15s; };
|
2024-02-18 20:54:43 +00:00
|
|
|
int GetCongestionLevel (bool longTerm) const;
|
2015-05-05 14:33:19 +00:00
|
|
|
size_t GetNumPeers () const { return m_Peers.size (); };
|
2023-02-03 20:59:56 +00:00
|
|
|
std::shared_ptr<const i2p::data::RouterInfo> GetRandomPeer (bool isHighBandwidth) const;
|
2015-03-17 19:19:38 +00:00
|
|
|
|
2020-03-01 10:25:50 +00:00
|
|
|
/** get a trusted first hop for restricted routes */
|
|
|
|
std::shared_ptr<const i2p::data::RouterInfo> GetRestrictedPeer() const;
|
|
|
|
/** do we want to use restricted routes? */
|
|
|
|
bool RoutesRestricted() const;
|
|
|
|
/** restrict routes to use only these router families for first hops */
|
2022-03-24 19:50:20 +00:00
|
|
|
void RestrictRoutesToFamilies(const std::set<std::string>& families);
|
2020-03-01 10:25:50 +00:00
|
|
|
/** restrict routes to use only these routers for first hops */
|
2024-02-09 20:24:48 +00:00
|
|
|
void RestrictRoutesToRouters(const std::set<i2p::data::IdentHash>& routers);
|
2016-10-28 16:50:26 +00:00
|
|
|
|
2020-03-01 10:25:50 +00:00
|
|
|
bool IsRestrictedPeer(const i2p::data::IdentHash & ident) const;
|
2017-05-29 05:28:16 +00:00
|
|
|
|
2021-05-04 18:59:25 +00:00
|
|
|
void PeerTest (bool ipv4 = true, bool ipv6 = true);
|
2017-05-29 05:28:16 +00:00
|
|
|
|
2020-10-12 15:27:25 +00:00
|
|
|
void SetCheckReserved (bool check) { m_CheckReserved = check; };
|
2024-01-11 20:39:42 +00:00
|
|
|
bool IsCheckReserved () const { return m_CheckReserved; };
|
|
|
|
bool IsInReservedRange (const boost::asio::ip::address& host) const;
|
2020-10-12 14:36:44 +00:00
|
|
|
|
2013-10-27 15:26:39 +00:00
|
|
|
private:
|
|
|
|
|
2013-11-29 12:52:09 +00:00
|
|
|
void Run ();
|
2015-01-14 21:37:03 +00:00
|
|
|
void RequestComplete (std::shared_ptr<const i2p::data::RouterInfo> r, const i2p::data::IdentHash& ident);
|
2015-12-31 16:21:01 +00:00
|
|
|
void HandleRequestComplete (std::shared_ptr<const i2p::data::RouterInfo> r, i2p::data::IdentHash ident);
|
2024-11-29 02:56:26 +00:00
|
|
|
std::shared_ptr<TransportSession> PostMessages (const i2p::data::IdentHash& ident, std::list<std::shared_ptr<i2p::I2NPMessage> >& msgs);
|
2024-04-27 12:18:49 +00:00
|
|
|
bool ConnectToPeer (const i2p::data::IdentHash& ident, std::shared_ptr<Peer> peer);
|
|
|
|
void SetPriority (std::shared_ptr<Peer> peer) const;
|
2017-05-29 05:28:16 +00:00
|
|
|
void HandlePeerCleanupTimer (const boost::system::error_code& ecode);
|
2016-11-13 14:14:05 +00:00
|
|
|
void HandlePeerTestTimer (const boost::system::error_code& ecode);
|
2022-12-12 05:12:23 +00:00
|
|
|
void HandleUpdateBandwidthTimer (const boost::system::error_code& ecode);
|
2024-02-18 20:54:43 +00:00
|
|
|
void UpdateBandwidthValues (int interval, uint32_t& in, uint32_t& out, uint32_t& transit);
|
2017-05-29 05:28:16 +00:00
|
|
|
|
2014-02-09 02:06:40 +00:00
|
|
|
void DetectExternalIP ();
|
2017-05-29 05:28:16 +00:00
|
|
|
|
2023-02-01 21:04:09 +00:00
|
|
|
template<typename Filter>
|
|
|
|
std::shared_ptr<const i2p::data::RouterInfo> GetRandomPeer (Filter filter) const;
|
2023-02-11 06:41:51 +00:00
|
|
|
|
2013-10-27 15:26:39 +00:00
|
|
|
private:
|
|
|
|
|
2020-10-11 21:51:40 +00:00
|
|
|
volatile bool m_IsOnline;
|
2020-10-12 15:27:25 +00:00
|
|
|
bool m_IsRunning, m_IsNAT, m_CheckReserved;
|
2017-05-29 05:28:16 +00:00
|
|
|
std::thread * m_Thread;
|
2024-11-25 15:08:27 +00:00
|
|
|
boost::asio::io_context * m_Service;
|
2024-11-25 21:00:06 +00:00
|
|
|
boost::asio::executor_work_guard<boost::asio::io_context::executor_type> * m_Work;
|
2022-12-13 20:25:16 +00:00
|
|
|
boost::asio::deadline_timer * m_PeerCleanupTimer, * m_PeerTestTimer, * m_UpdateBandwidthTimer;
|
2013-10-27 15:26:39 +00:00
|
|
|
|
2022-03-11 21:17:44 +00:00
|
|
|
SSU2Server * m_SSU2Server;
|
2018-07-13 19:59:28 +00:00
|
|
|
NTCP2Server * m_NTCP2Server;
|
2016-01-15 21:23:03 +00:00
|
|
|
mutable std::mutex m_PeersMutex;
|
2024-04-27 12:18:49 +00:00
|
|
|
std::unordered_map<i2p::data::IdentHash, std::shared_ptr<Peer> > m_Peers;
|
2017-05-29 05:28:16 +00:00
|
|
|
|
2020-06-30 00:02:09 +00:00
|
|
|
X25519KeysPairSupplier m_X25519KeysPairSupplier;
|
2015-03-17 19:19:38 +00:00
|
|
|
|
2017-05-02 18:20:00 +00:00
|
|
|
std::atomic<uint64_t> m_TotalSentBytes, m_TotalReceivedBytes, m_TotalTransitTransmittedBytes;
|
2022-12-12 05:12:23 +00:00
|
|
|
|
2024-02-18 20:54:43 +00:00
|
|
|
TrafficSample m_TrafficSamples[TRAFFIC_SAMPLE_COUNT];
|
|
|
|
int m_TrafficSamplePtr;
|
|
|
|
|
2022-12-12 05:12:23 +00:00
|
|
|
// Bandwidth per second
|
|
|
|
uint32_t m_InBandwidth, m_OutBandwidth, m_TransitBandwidth;
|
2024-02-18 20:54:43 +00:00
|
|
|
// Bandwidth during last 15 seconds
|
2022-12-12 05:12:23 +00:00
|
|
|
uint32_t m_InBandwidth15s, m_OutBandwidth15s, m_TransitBandwidth15s;
|
2024-02-18 20:54:43 +00:00
|
|
|
// Bandwidth during last 5 minutes
|
|
|
|
uint32_t m_InBandwidth5m, m_OutBandwidth5m, m_TransitBandwidth5m;
|
2015-03-17 19:19:38 +00:00
|
|
|
|
2016-11-15 19:11:55 +00:00
|
|
|
/** which router families to trust for first hops */
|
2022-03-24 19:50:20 +00:00
|
|
|
std::vector<i2p::data::FamilyID> m_TrustedFamilies;
|
2016-11-15 19:11:55 +00:00
|
|
|
mutable std::mutex m_FamilyMutex;
|
2016-10-28 16:50:26 +00:00
|
|
|
|
2016-11-15 19:11:55 +00:00
|
|
|
/** which routers for first hop to trust */
|
|
|
|
std::vector<i2p::data::IdentHash> m_TrustedRouters;
|
|
|
|
mutable std::mutex m_TrustedRoutersMutex;
|
|
|
|
|
2017-05-29 05:28:16 +00:00
|
|
|
i2p::I2NPMessagesHandler m_LoopbackHandler;
|
|
|
|
|
2013-12-10 13:10:49 +00:00
|
|
|
public:
|
|
|
|
|
|
|
|
// for HTTP only
|
2018-07-13 19:59:28 +00:00
|
|
|
const NTCP2Server * GetNTCP2Server () const { return m_NTCP2Server; };
|
2022-03-27 23:29:50 +00:00
|
|
|
const SSU2Server * GetSSU2Server () const { return m_SSU2Server; };
|
2015-01-14 18:21:41 +00:00
|
|
|
const decltype(m_Peers)& GetPeers () const { return m_Peers; };
|
2017-05-29 05:28:16 +00:00
|
|
|
};
|
2013-10-27 15:26:39 +00:00
|
|
|
|
|
|
|
extern Transports transports;
|
2022-09-24 01:27:11 +00:00
|
|
|
|
2022-10-10 12:43:37 +00:00
|
|
|
void InitAddressFromIface ();
|
2022-09-24 01:27:11 +00:00
|
|
|
void InitTransports ();
|
2017-05-29 05:28:16 +00:00
|
|
|
}
|
2014-10-21 16:25:53 +00:00
|
|
|
}
|
2013-10-27 15:26:39 +00:00
|
|
|
|
|
|
|
#endif
|