|
|
|
#ifndef DATAGRAM_H__
|
|
|
|
#define DATAGRAM_H__
|
|
|
|
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include <memory>
|
|
|
|
#include <functional>
|
|
|
|
#include <map>
|
|
|
|
#include "Base.h"
|
|
|
|
#include "Identity.h"
|
|
|
|
#include "LeaseSet.h"
|
|
|
|
#include "I2NPProtocol.h"
|
|
|
|
#include "Garlic.h"
|
|
|
|
|
|
|
|
namespace i2p
|
|
|
|
{
|
|
|
|
namespace client
|
|
|
|
{
|
|
|
|
class ClientDestination;
|
|
|
|
}
|
|
|
|
namespace datagram
|
|
|
|
{
|
|
|
|
|
|
|
|
// seconds interval for cleanup timer
|
|
|
|
const int DATAGRAM_SESSION_CLEANUP_INTERVAL = 3;
|
|
|
|
// milliseconds for max session idle time (10 minutes)
|
|
|
|
const uint64_t DATAGRAM_SESSION_MAX_IDLE = 3600 * 1000;
|
|
|
|
|
|
|
|
|
|
|
|
class DatagramSession
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
DatagramSession(i2p::client::ClientDestination * localDestination,
|
|
|
|
const i2p::data::IdentHash & remoteIdent);
|
|
|
|
|
|
|
|
/** send an i2np message to remote endpoint for this session */
|
|
|
|
void SendMsg(std::shared_ptr<I2NPMessage> msg);
|
|
|
|
/** get the last time in milliseconds for when we used this datagram session */
|
|
|
|
uint64_t LastActivity() const { return m_LastUse; }
|
|
|
|
private:
|
|
|
|
|
|
|
|
/** get next usable routing path, try reusing outbound tunnels */
|
|
|
|
std::shared_ptr<i2p::garlic::GarlicRoutingPath> GetNextRoutingPath();
|
|
|
|
/**
|
|
|
|
* mark current routing path as invalid and clear it
|
|
|
|
* if the outbound tunnel we were using was okay don't use the IBGW in the routing path's lease next time
|
|
|
|
*/
|
|
|
|
void ResetRoutingPath();
|
|
|
|
|
|
|
|
/** get next usable lease, does not fetch or update if expired or have no lease set */
|
|
|
|
std::shared_ptr<const i2p::data::Lease> GetNextLease();
|
|
|
|
|
|
|
|
void HandleSend(std::shared_ptr<I2NPMessage> msg);
|
|
|
|
void HandleGotLeaseSet(std::shared_ptr<const i2p::data::LeaseSet> remoteIdent,
|
|
|
|
std::shared_ptr<I2NPMessage> msg);
|
|
|
|
void UpdateLeaseSet(std::shared_ptr<I2NPMessage> msg=nullptr);
|
|
|
|
|
|
|
|
private:
|
|
|
|
i2p::client::ClientDestination * m_LocalDestination;
|
|
|
|
i2p::data::IdentHash m_RemoteIdentity;
|
|
|
|
std::shared_ptr<i2p::garlic::GarlicRoutingSession> m_RoutingSession;
|
|
|
|
// Ident hash of IBGW that are invalid
|
|
|
|
std::vector<i2p::data::IdentHash> m_InvalidIBGW;
|
|
|
|
std::shared_ptr<const i2p::data::LeaseSet> m_RemoteLeaseSet;
|
|
|
|
uint64_t m_LastUse;
|
|
|
|
};
|
|
|
|
|
|
|
|
const size_t MAX_DATAGRAM_SIZE = 32768;
|
|
|
|
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;
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
DatagramDestination (std::shared_ptr<i2p::client::ClientDestination> owner);
|
|
|
|
~DatagramDestination ();
|
|
|
|
|
|
|
|
void SendDatagramTo (const uint8_t * payload, size_t len, const i2p::data::IdentHash& ident, uint16_t fromPort = 0, uint16_t toPort = 0);
|
|
|
|
void HandleDataMessagePayload (uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len);
|
|
|
|
|
|
|
|
void SetReceiver (const Receiver& receiver) { m_Receiver = receiver; };
|
|
|
|
void ResetReceiver () { m_Receiver = nullptr; };
|
|
|
|
|
|
|
|
void SetReceiver (const Receiver& receiver, uint16_t port) { m_ReceiversByPorts[port] = receiver; };
|
|
|
|
void ResetReceiver (uint16_t port) { m_ReceiversByPorts.erase (port); };
|
|
|
|
|
|
|
|
private:
|
|
|
|
// clean up after next tick
|
|
|
|
void ScheduleCleanup();
|
|
|
|
|
|
|
|
// clean up stale sessions and expire tags
|
|
|
|
void HandleCleanUp(const boost::system::error_code & ecode);
|
|
|
|
|
|
|
|
std::shared_ptr<DatagramSession> ObtainSession(const i2p::data::IdentHash & ident);
|
|
|
|
|
|
|
|
std::shared_ptr<I2NPMessage> CreateDataMessage (const uint8_t * payload, size_t len, uint16_t fromPort, uint16_t toPort);
|
|
|
|
|
|
|
|
void HandleDatagram (uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len);
|
|
|
|
|
|
|
|
private:
|
|
|
|
i2p::client::ClientDestination * m_Owner;
|
|
|
|
boost::asio::deadline_timer m_CleanupTimer;
|
|
|
|
Receiver m_Receiver; // default
|
|
|
|
std::mutex m_SessionsMutex;
|
|
|
|
std::map<i2p::data::IdentHash, std::shared_ptr<DatagramSession> > m_Sessions;
|
|
|
|
std::map<uint16_t, Receiver> m_ReceiversByPorts;
|
|
|
|
|
|
|
|
i2p::data::GzipInflator m_Inflator;
|
|
|
|
i2p::data::GzipDeflator m_Deflator;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|