1
0
mirror of https://github.com/PurpleI2P/i2pd.git synced 2025-01-25 19:34:13 +00:00
i2pd/libi2pd/Datagram.h

180 lines
6.3 KiB
C
Raw Normal View History

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