2020-05-22 15:18:41 +02:00
|
|
|
/*
|
2025-01-09 02:52:38 +01:00
|
|
|
* Copyright (c) 2013-2025, The PurpleI2P Project
|
2020-05-22 15:18:41 +02: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-10-22 21:30:25 +02:00
|
|
|
#ifndef DATAGRAM_H__
|
|
|
|
#define DATAGRAM_H__
|
|
|
|
|
|
|
|
#include <inttypes.h>
|
2015-01-29 03:37:08 +01:00
|
|
|
#include <memory>
|
2014-10-31 21:44:44 +01:00
|
|
|
#include <functional>
|
2015-04-04 02:34:37 +02:00
|
|
|
#include <map>
|
2020-05-22 03:54:00 +02:00
|
|
|
#include <vector>
|
2015-11-03 15:15:49 +01:00
|
|
|
#include "Base.h"
|
2022-10-25 21:30:12 +02:00
|
|
|
#include "Gzip.h"
|
2014-10-31 21:44:44 +01:00
|
|
|
#include "Identity.h"
|
2014-10-23 22:56:50 +02:00
|
|
|
#include "LeaseSet.h"
|
|
|
|
#include "I2NPProtocol.h"
|
2016-08-27 19:17:34 +02:00
|
|
|
#include "Garlic.h"
|
2014-10-22 21:30:25 +02:00
|
|
|
|
|
|
|
namespace i2p
|
|
|
|
{
|
|
|
|
namespace client
|
|
|
|
{
|
2016-05-25 22:18:02 +02:00
|
|
|
class ClientDestination;
|
2014-10-22 21:30:25 +02:00
|
|
|
}
|
|
|
|
namespace datagram
|
|
|
|
{
|
2017-04-08 18:51:35 +02:00
|
|
|
// milliseconds for max session idle time
|
2016-09-03 17:46:47 +02:00
|
|
|
const uint64_t DATAGRAM_SESSION_MAX_IDLE = 10 * 60 * 1000;
|
|
|
|
// milliseconds interval a routing path is used before switching
|
2016-10-06 19:41:18 +02:00
|
|
|
const uint64_t DATAGRAM_SESSION_PATH_SWITCH_INTERVAL = 20 * 60 * 1000;
|
2016-09-03 17:46:47 +02:00
|
|
|
// milliseconds before lease expire should we try switching leases
|
2017-04-09 14:52:42 +02:00
|
|
|
const uint64_t DATAGRAM_SESSION_LEASE_HANDOVER_WINDOW = 30 * 1000;
|
2016-09-03 17:46:47 +02:00
|
|
|
// milliseconds fudge factor for leases handover
|
|
|
|
const uint64_t DATAGRAM_SESSION_LEASE_HANDOVER_FUDGE = 1000;
|
2016-10-10 14:30:33 +02:00
|
|
|
// milliseconds minimum time between path switches
|
|
|
|
const uint64_t DATAGRAM_SESSION_PATH_MIN_LIFETIME = 5 * 1000;
|
2020-03-01 11:25:50 +01:00
|
|
|
// max 64 messages buffered in send queue for each datagram session
|
|
|
|
const size_t DATAGRAM_SEND_QUEUE_MAX_SIZE = 64;
|
2024-08-21 02:59:41 +02:00
|
|
|
const uint64_t DATAGRAM_MAX_FLUSH_INTERVAL = 5; // in milliseconds
|
2025-01-09 02:52:38 +01:00
|
|
|
const int DATAGRAM_SESSION_ACK_REQUEST_INTERVAL = 5500; // in milliseconds
|
2017-04-08 18:51:35 +02:00
|
|
|
|
2017-01-16 13:54:56 +01:00
|
|
|
class DatagramSession : public std::enable_shared_from_this<DatagramSession>
|
2016-09-03 17:46:47 +02:00
|
|
|
{
|
2017-01-17 18:13:56 +01:00
|
|
|
|
2020-03-01 11:25:50 +01:00
|
|
|
public:
|
|
|
|
|
|
|
|
DatagramSession(std::shared_ptr<i2p::client::ClientDestination> localDestination, const i2p::data::IdentHash & remoteIdent);
|
|
|
|
|
|
|
|
void Start ();
|
|
|
|
void Stop ();
|
2016-12-12 19:40:24 +01:00
|
|
|
|
|
|
|
|
2020-03-01 11:25:50 +01:00
|
|
|
/** @brief ack the garlic routing path */
|
|
|
|
void Ack();
|
2016-09-03 17:46:47 +02:00
|
|
|
|
2020-03-01 11:25:50 +01:00
|
|
|
/** send an i2np message to remote endpoint for this session */
|
|
|
|
void SendMsg(std::shared_ptr<I2NPMessage> msg);
|
2020-06-09 22:26:45 +02:00
|
|
|
void FlushSendQueue();
|
2020-03-01 11:25:50 +01:00
|
|
|
/** get the last time in milliseconds for when we used this datagram session */
|
|
|
|
uint64_t LastActivity() const { return m_LastUse; }
|
2016-12-12 19:40:24 +01:00
|
|
|
|
2020-05-17 22:49:31 +02:00
|
|
|
bool IsRatchets () const { return m_RoutingSession && m_RoutingSession->IsRatchets (); }
|
2020-03-01 11:25:50 +01:00
|
|
|
|
2016-09-03 19:58:34 +02:00
|
|
|
struct Info
|
|
|
|
{
|
2016-09-03 22:54:39 +02:00
|
|
|
std::shared_ptr<const i2p::data::IdentHash> IBGW;
|
|
|
|
std::shared_ptr<const i2p::data::IdentHash> OBEP;
|
2016-09-03 19:58:34 +02:00
|
|
|
const uint64_t activity;
|
2017-04-08 18:51:35 +02:00
|
|
|
|
2020-03-01 11:25:50 +01:00
|
|
|
Info() : IBGW(nullptr), OBEP(nullptr), activity(0) {}
|
|
|
|
Info(const uint8_t * ibgw, const uint8_t * obep, const uint64_t a) :
|
|
|
|
activity(a) {
|
|
|
|
if(ibgw) IBGW = std::make_shared<i2p::data::IdentHash>(ibgw);
|
|
|
|
else IBGW = nullptr;
|
|
|
|
if(obep) OBEP = std::make_shared<i2p::data::IdentHash>(obep);
|
|
|
|
else OBEP = nullptr;
|
|
|
|
}
|
|
|
|
};
|
2016-09-03 19:58:34 +02:00
|
|
|
|
2020-03-01 11:25:50 +01:00
|
|
|
Info GetSessionInfo() const;
|
2016-09-03 19:58:34 +02:00
|
|
|
|
2020-03-01 11:25:50 +01:00
|
|
|
private:
|
2016-09-03 17:46:47 +02:00
|
|
|
|
2020-03-01 11:25:50 +01:00
|
|
|
std::shared_ptr<i2p::garlic::GarlicRoutingPath> GetSharedRoutingPath();
|
2017-04-08 18:51:35 +02:00
|
|
|
|
2020-03-01 11:25:50 +01:00
|
|
|
void HandleLeaseSetUpdated(std::shared_ptr<i2p::data::LeaseSet> ls);
|
2016-09-03 17:46:47 +02:00
|
|
|
|
2020-03-01 11:25:50 +01:00
|
|
|
private:
|
|
|
|
|
|
|
|
std::shared_ptr<i2p::client::ClientDestination> m_LocalDestination;
|
|
|
|
i2p::data::IdentHash m_RemoteIdent;
|
|
|
|
std::shared_ptr<const i2p::data::LeaseSet> m_RemoteLeaseSet;
|
|
|
|
std::shared_ptr<i2p::garlic::GarlicRoutingSession> m_RoutingSession;
|
2020-06-14 17:16:08 +02:00
|
|
|
std::vector<std::shared_ptr<i2p::garlic::GarlicRoutingSession> > m_PendingRoutingSessions;
|
2020-03-01 11:25:50 +01:00
|
|
|
std::vector<std::shared_ptr<I2NPMessage> > m_SendQueue;
|
2024-08-21 02:59:41 +02:00
|
|
|
uint64_t m_LastUse, m_LastFlush; // milliseconds
|
2020-03-01 11:25:50 +01:00
|
|
|
bool m_RequestingLS;
|
2016-09-03 17:46:47 +02:00
|
|
|
};
|
2017-01-16 13:54:56 +01:00
|
|
|
|
|
|
|
typedef std::shared_ptr<DatagramSession> DatagramSession_ptr;
|
|
|
|
|
2017-04-08 18:51:35 +02:00
|
|
|
const size_t MAX_DATAGRAM_SIZE = 32768;
|
2014-10-22 21:30:25 +02:00
|
|
|
class DatagramDestination
|
|
|
|
{
|
2015-03-03 21:31:49 +01:00
|
|
|
typedef std::function<void (const i2p::data::IdentityEx& from, uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len)> Receiver;
|
2019-07-10 03:33:55 +02:00
|
|
|
typedef std::function<void (uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len)> RawReceiver;
|
2014-10-31 21:44:44 +01:00
|
|
|
|
2014-10-22 21:30:25 +02:00
|
|
|
public:
|
|
|
|
|
2020-03-01 11:25:50 +01:00
|
|
|
DatagramDestination (std::shared_ptr<i2p::client::ClientDestination> owner, bool gzip);
|
2017-04-08 18:51:35 +02:00
|
|
|
~DatagramDestination ();
|
2014-10-22 21:30:25 +02:00
|
|
|
|
2019-07-10 03:33:55 +02:00
|
|
|
void SendDatagramTo (const uint8_t * payload, size_t len, const i2p::data::IdentHash & ident, uint16_t fromPort = 0, uint16_t toPort = 0);
|
|
|
|
void SendRawDatagramTo (const uint8_t * payload, size_t len, const i2p::data::IdentHash & ident, uint16_t fromPort = 0, uint16_t toPort = 0);
|
2020-06-11 03:19:37 +02:00
|
|
|
// TODO: implement calls from other thread from SAM
|
2021-11-27 21:30:35 +01:00
|
|
|
|
2020-06-10 01:20:24 +02:00
|
|
|
std::shared_ptr<DatagramSession> GetSession(const i2p::data::IdentHash & ident);
|
|
|
|
void SendDatagram (std::shared_ptr<DatagramSession> session, const uint8_t * payload, size_t len, uint16_t fromPort, uint16_t toPort);
|
|
|
|
void SendRawDatagram (std::shared_ptr<DatagramSession> session, const uint8_t * payload, size_t len, uint16_t fromPort, uint16_t toPort);
|
|
|
|
void FlushSendQueue (std::shared_ptr<DatagramSession> session);
|
2021-11-27 21:30:35 +01:00
|
|
|
|
2019-07-10 03:33:55 +02:00
|
|
|
void HandleDataMessagePayload (uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len, bool isRaw = false);
|
2020-03-01 11:25:50 +01:00
|
|
|
|
2014-10-31 21:44:44 +01:00
|
|
|
|
2024-01-25 02:05:58 +01:00
|
|
|
void SetReceiver (const Receiver& receiver, uint16_t port);
|
|
|
|
void ResetReceiver (uint16_t port);
|
2016-09-03 19:58:34 +02:00
|
|
|
|
2024-01-25 02:05:58 +01:00
|
|
|
void SetRawReceiver (const RawReceiver& receiver, uint16_t port);
|
|
|
|
void ResetRawReceiver (uint16_t port);
|
2020-03-01 11:25:50 +01:00
|
|
|
|
2016-09-03 19:58:34 +02:00
|
|
|
std::shared_ptr<DatagramSession::Info> GetInfoForRemote(const i2p::data::IdentHash & remote);
|
2017-04-08 18:51:35 +02:00
|
|
|
|
2016-09-08 16:16:42 +02:00
|
|
|
// clean up stale sessions
|
|
|
|
void CleanUp ();
|
|
|
|
|
2014-10-23 22:56:50 +02:00
|
|
|
private:
|
2017-04-08 18:51:35 +02:00
|
|
|
|
2020-03-01 11:25:50 +01:00
|
|
|
std::shared_ptr<DatagramSession> ObtainSession(const i2p::data::IdentHash & ident);
|
2017-04-08 18:51:35 +02:00
|
|
|
|
2020-03-01 11:25:50 +01:00
|
|
|
std::shared_ptr<I2NPMessage> CreateDataMessage (const std::vector<std::pair<const uint8_t *, size_t> >& payloads,
|
2020-05-17 22:49:31 +02:00
|
|
|
uint16_t fromPort, uint16_t toPort, bool isRaw = false, bool checksum = true);
|
2016-08-27 19:17:34 +02:00
|
|
|
|
2016-10-09 16:55:55 +02:00
|
|
|
void HandleDatagram (uint16_t fromPort, uint16_t toPort, uint8_t *const& buf, size_t len);
|
2019-07-10 03:33:55 +02:00
|
|
|
void HandleRawDatagram (uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len);
|
2020-03-01 11:25:50 +01:00
|
|
|
|
2016-09-03 16:24:06 +02:00
|
|
|
Receiver FindReceiver(uint16_t port);
|
2024-01-25 02:05:58 +01:00
|
|
|
RawReceiver FindRawReceiver(uint16_t port);
|
2017-04-08 18:51:35 +02:00
|
|
|
|
2014-10-22 21:30:25 +02:00
|
|
|
private:
|
2019-07-10 03:33:55 +02:00
|
|
|
|
|
|
|
std::shared_ptr<i2p::client::ClientDestination> m_Owner;
|
2024-01-25 02:05:58 +01:00
|
|
|
|
2016-09-03 16:24:06 +02:00
|
|
|
std::mutex m_SessionsMutex;
|
2017-01-16 13:54:56 +01:00
|
|
|
std::map<i2p::data::IdentHash, DatagramSession_ptr > m_Sessions;
|
2024-01-25 02:05:58 +01:00
|
|
|
|
|
|
|
Receiver m_DefaultReceiver;
|
|
|
|
RawReceiver m_DefaultRawReceiver;
|
|
|
|
uint16_t m_DefaultReceiverPort;
|
|
|
|
uint16_t m_DefaultRawReceiverPort;
|
2016-09-03 16:24:06 +02:00
|
|
|
std::mutex m_ReceiversMutex;
|
2024-01-25 02:05:58 +01:00
|
|
|
std::mutex m_RawReceiversMutex;
|
|
|
|
std::unordered_map<uint16_t, Receiver> m_ReceiversByPorts;
|
|
|
|
std::unordered_map<uint16_t, RawReceiver> m_RawReceiversByPorts;
|
2015-11-03 15:15:49 +01:00
|
|
|
|
2024-01-25 02:05:58 +01:00
|
|
|
bool m_Gzip; // gzip compression of data messages
|
2015-11-03 15:15:49 +01:00
|
|
|
i2p::data::GzipInflator m_Inflator;
|
2021-09-14 13:48:21 +02:00
|
|
|
std::unique_ptr<i2p::data::GzipDeflator> m_Deflator;
|
2020-05-22 03:54:00 +02:00
|
|
|
std::vector<uint8_t> m_From, m_Signature;
|
2020-06-11 03:19:37 +02:00
|
|
|
i2p::util::MemoryPool<I2NPMessageBuffer<I2NP_MAX_MESSAGE_SIZE> > m_I2NPMsgsPool;
|
2017-04-08 18:51:35 +02:00
|
|
|
};
|
2014-10-22 21:30:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|