2014-10-05 16:54:59 +04:00
|
|
|
#ifndef DESTINATION_H__
|
|
|
|
#define DESTINATION_H__
|
|
|
|
|
|
|
|
#include <thread>
|
|
|
|
#include <mutex>
|
2014-11-23 19:33:58 +03:00
|
|
|
#include <memory>
|
2014-11-30 18:51:22 +03:00
|
|
|
#include <map>
|
2014-12-26 00:47:15 +03:00
|
|
|
#include <set>
|
2014-11-30 18:51:22 +03:00
|
|
|
#include <string>
|
2014-12-27 03:09:44 +03:00
|
|
|
#include <functional>
|
2016-10-19 17:23:02 +03:00
|
|
|
#ifdef I2LUA
|
2016-08-30 20:27:57 +03:00
|
|
|
#include <future>
|
2016-10-19 17:23:02 +03:00
|
|
|
#endif
|
2014-12-26 00:47:15 +03:00
|
|
|
#include <boost/asio.hpp>
|
2014-10-05 16:54:59 +04:00
|
|
|
#include "Identity.h"
|
|
|
|
#include "TunnelPool.h"
|
2015-11-03 17:15:49 +03:00
|
|
|
#include "Crypto.h"
|
2014-10-13 00:22:14 +04:00
|
|
|
#include "LeaseSet.h"
|
2014-10-07 04:18:18 +04:00
|
|
|
#include "Garlic.h"
|
2014-12-26 00:47:15 +03:00
|
|
|
#include "NetDb.h"
|
2014-10-05 16:54:59 +04:00
|
|
|
#include "Streaming.h"
|
2014-10-22 23:30:25 +04:00
|
|
|
#include "Datagram.h"
|
2014-10-05 16:54:59 +04:00
|
|
|
|
|
|
|
namespace i2p
|
|
|
|
{
|
2014-10-16 20:37:39 +04:00
|
|
|
namespace client
|
2014-10-05 16:54:59 +04:00
|
|
|
{
|
2014-10-22 22:01:23 +04:00
|
|
|
const uint8_t PROTOCOL_TYPE_STREAMING = 6;
|
|
|
|
const uint8_t PROTOCOL_TYPE_DATAGRAM = 17;
|
|
|
|
const uint8_t PROTOCOL_TYPE_RAW = 18;
|
2014-11-28 21:01:35 +03:00
|
|
|
const int PUBLISH_CONFIRMATION_TIMEOUT = 5; // in seconds
|
2016-02-11 22:45:33 +03:00
|
|
|
const int PUBLISH_VERIFICATION_TIMEOUT = 10; // in seconds after successfull publish
|
|
|
|
const int PUBLISH_REGULAR_VERIFICATION_INTERNAL = 100; // in seconds periodically
|
2014-12-26 00:47:15 +03:00
|
|
|
const int LEASESET_REQUEST_TIMEOUT = 5; // in seconds
|
|
|
|
const int MAX_LEASESET_REQUEST_TIMEOUT = 40; // in seconds
|
2016-01-26 06:10:06 +03:00
|
|
|
const int DESTINATION_CLEANUP_TIMEOUT = 3; // in minutes
|
2015-12-17 10:58:09 +03:00
|
|
|
const unsigned int MAX_NUM_FLOODFILLS_PER_REQUEST = 7;
|
2014-12-26 00:47:15 +03:00
|
|
|
|
2014-11-30 18:51:22 +03:00
|
|
|
// I2CP
|
|
|
|
const char I2CP_PARAM_INBOUND_TUNNEL_LENGTH[] = "inbound.length";
|
|
|
|
const int DEFAULT_INBOUND_TUNNEL_LENGTH = 3;
|
|
|
|
const char I2CP_PARAM_OUTBOUND_TUNNEL_LENGTH[] = "outbound.length";
|
|
|
|
const int DEFAULT_OUTBOUND_TUNNEL_LENGTH = 3;
|
2015-05-05 19:32:13 +03:00
|
|
|
const char I2CP_PARAM_INBOUND_TUNNELS_QUANTITY[] = "inbound.quantity";
|
|
|
|
const int DEFAULT_INBOUND_TUNNELS_QUANTITY = 5;
|
|
|
|
const char I2CP_PARAM_OUTBOUND_TUNNELS_QUANTITY[] = "outbound.quantity";
|
|
|
|
const int DEFAULT_OUTBOUND_TUNNELS_QUANTITY = 5;
|
2015-06-10 22:32:55 +03:00
|
|
|
const char I2CP_PARAM_EXPLICIT_PEERS[] = "explicitPeers";
|
2015-01-02 02:56:26 +03:00
|
|
|
const int STREAM_REQUEST_TIMEOUT = 60; //in seconds
|
2016-01-24 04:52:21 +03:00
|
|
|
const char I2CP_PARAM_TAGS_TO_SEND[] = "crypto.tagsToSend";
|
|
|
|
const int DEFAULT_TAGS_TO_SEND = 40;
|
2016-02-29 22:44:15 +03:00
|
|
|
|
2015-01-07 22:44:24 +03:00
|
|
|
typedef std::function<void (std::shared_ptr<i2p::stream::Stream> stream)> StreamRequestComplete;
|
|
|
|
|
2016-05-23 17:33:01 +03:00
|
|
|
class LeaseSetDestination: public i2p::garlic::GarlicDestination,
|
|
|
|
public std::enable_shared_from_this<LeaseSetDestination>
|
2014-10-05 16:54:59 +04:00
|
|
|
{
|
2015-04-07 19:02:25 +03:00
|
|
|
typedef std::function<void (std::shared_ptr<i2p::data::LeaseSet> leaseSet)> RequestComplete;
|
|
|
|
// leaseSet = nullptr means not found
|
2014-12-26 00:47:15 +03:00
|
|
|
struct LeaseSetRequest
|
|
|
|
{
|
|
|
|
LeaseSetRequest (boost::asio::io_service& service): requestTime (0), requestTimeoutTimer (service) {};
|
|
|
|
std::set<i2p::data::IdentHash> excluded;
|
|
|
|
uint64_t requestTime;
|
|
|
|
boost::asio::deadline_timer requestTimeoutTimer;
|
2014-12-27 03:09:44 +03:00
|
|
|
RequestComplete requestComplete;
|
2016-02-27 00:17:29 +03:00
|
|
|
std::shared_ptr<i2p::tunnel::OutboundTunnel> outboundTunnel;
|
|
|
|
std::shared_ptr<i2p::tunnel::InboundTunnel> replyTunnel;
|
2014-12-26 00:47:15 +03:00
|
|
|
};
|
|
|
|
|
2015-01-02 15:35:38 +03:00
|
|
|
|
2014-10-05 16:54:59 +04:00
|
|
|
public:
|
|
|
|
|
2016-05-23 17:33:01 +03:00
|
|
|
LeaseSetDestination (bool isPublic, const std::map<std::string, std::string> * params = nullptr);
|
|
|
|
~LeaseSetDestination ();
|
2014-10-05 16:54:59 +04:00
|
|
|
|
2016-05-23 21:31:22 +03:00
|
|
|
virtual bool Start ();
|
|
|
|
virtual bool Stop ();
|
2014-10-13 19:21:57 +04:00
|
|
|
bool IsRunning () const { return m_IsRunning; };
|
2014-12-16 06:50:11 +03:00
|
|
|
boost::asio::io_service& GetService () { return m_Service; };
|
2015-01-20 06:28:13 +03:00
|
|
|
std::shared_ptr<i2p::tunnel::TunnelPool> GetTunnelPool () { return m_Pool; };
|
2016-02-08 03:45:06 +03:00
|
|
|
bool IsReady () const { return m_LeaseSet && !m_LeaseSet->IsExpired () && m_Pool->GetOutboundTunnels ().size () > 0; };
|
2015-01-27 19:27:58 +03:00
|
|
|
std::shared_ptr<const i2p::data::LeaseSet> FindLeaseSet (const i2p::data::IdentHash& ident);
|
2014-12-27 03:09:44 +03:00
|
|
|
bool RequestDestination (const i2p::data::IdentHash& dest, RequestComplete requestComplete = nullptr);
|
2016-08-08 18:53:38 +03:00
|
|
|
void CancelDestinationRequest (const i2p::data::IdentHash& dest, bool notify = true);
|
2016-05-26 22:53:32 +03:00
|
|
|
|
2014-10-08 05:08:00 +04:00
|
|
|
// implements GarlicDestination
|
2016-05-25 22:10:28 +03:00
|
|
|
std::shared_ptr<const i2p::data::LocalLeaseSet> GetLeaseSet ();
|
2015-04-05 19:54:15 +03:00
|
|
|
std::shared_ptr<i2p::tunnel::TunnelPool> GetTunnelPool () const { return m_Pool; }
|
2015-02-06 02:53:43 +03:00
|
|
|
void HandleI2NPMessage (const uint8_t * buf, size_t len, std::shared_ptr<i2p::tunnel::InboundTunnel> from);
|
2014-10-08 05:08:00 +04:00
|
|
|
|
2014-10-08 22:17:17 +04:00
|
|
|
// override GarlicDestination
|
2014-12-08 23:36:00 +03:00
|
|
|
bool SubmitSessionKey (const uint8_t * key, const uint8_t * tag);
|
2015-06-16 17:14:14 +03:00
|
|
|
void ProcessGarlicMessage (std::shared_ptr<I2NPMessage> msg);
|
|
|
|
void ProcessDeliveryStatusMessage (std::shared_ptr<I2NPMessage> msg);
|
2014-10-08 22:17:17 +04:00
|
|
|
void SetLeaseSetUpdated ();
|
|
|
|
|
2016-05-23 21:31:22 +03:00
|
|
|
protected:
|
2014-10-16 20:37:39 +04:00
|
|
|
|
2016-05-29 16:33:50 +03:00
|
|
|
void SetLeaseSet (i2p::data::LocalLeaseSet * newLeaseSet);
|
2016-09-08 17:16:42 +03:00
|
|
|
virtual void CleanupDestination () {}; // additional clean up in derived classes
|
2016-05-23 21:31:22 +03:00
|
|
|
// I2CP
|
|
|
|
virtual void HandleDataMessage (const uint8_t * buf, size_t len) = 0;
|
2016-05-29 16:33:50 +03:00
|
|
|
virtual void CreateNewLeaseSet (std::vector<std::shared_ptr<i2p::tunnel::InboundTunnel> > tunnels) = 0;
|
2016-05-23 21:31:22 +03:00
|
|
|
|
2014-10-16 20:37:39 +04:00
|
|
|
private:
|
|
|
|
|
|
|
|
void Run ();
|
2014-10-05 16:54:59 +04:00
|
|
|
void UpdateLeaseSet ();
|
2014-11-28 21:01:35 +03:00
|
|
|
void Publish ();
|
|
|
|
void HandlePublishConfirmationTimer (const boost::system::error_code& ecode);
|
2016-02-11 22:45:33 +03:00
|
|
|
void HandlePublishVerificationTimer (const boost::system::error_code& ecode);
|
2014-12-26 00:47:15 +03:00
|
|
|
void HandleDatabaseStoreMessage (const uint8_t * buf, size_t len);
|
|
|
|
void HandleDatabaseSearchReplyMessage (const uint8_t * buf, size_t len);
|
2015-06-16 17:14:14 +03:00
|
|
|
void HandleDeliveryStatusMessage (std::shared_ptr<I2NPMessage> msg);
|
2014-10-05 16:54:59 +04:00
|
|
|
|
2014-12-27 03:09:44 +03:00
|
|
|
void RequestLeaseSet (const i2p::data::IdentHash& dest, RequestComplete requestComplete);
|
2015-12-13 18:51:43 +03:00
|
|
|
bool SendLeaseSetRequest (const i2p::data::IdentHash& dest, std::shared_ptr<const i2p::data::RouterInfo> nextFloodfill, std::shared_ptr<LeaseSetRequest> request);
|
2014-12-26 00:47:15 +03:00
|
|
|
void HandleRequestTimoutTimer (const boost::system::error_code& ecode, const i2p::data::IdentHash& dest);
|
2015-01-23 20:48:25 +03:00
|
|
|
void HandleCleanupTimer (const boost::system::error_code& ecode);
|
2016-05-23 17:33:01 +03:00
|
|
|
void CleanupRemoteLeaseSets ();
|
2016-05-23 21:31:22 +03:00
|
|
|
|
2014-10-05 16:54:59 +04:00
|
|
|
private:
|
|
|
|
|
2015-01-03 23:20:11 +03:00
|
|
|
volatile bool m_IsRunning;
|
2014-10-09 18:05:28 +04:00
|
|
|
std::thread * m_Thread;
|
2014-12-16 06:50:11 +03:00
|
|
|
boost::asio::io_service m_Service;
|
|
|
|
boost::asio::io_service::work m_Work;
|
2016-07-22 16:56:17 +03:00
|
|
|
mutable std::mutex m_RemoteLeaseSetsMutex;
|
2015-01-27 19:27:58 +03:00
|
|
|
std::map<i2p::data::IdentHash, std::shared_ptr<i2p::data::LeaseSet> > m_RemoteLeaseSets;
|
2015-12-13 18:51:43 +03:00
|
|
|
std::map<i2p::data::IdentHash, std::shared_ptr<LeaseSetRequest> > m_LeaseSetRequests;
|
2014-10-16 20:37:39 +04:00
|
|
|
|
2015-01-20 06:28:13 +03:00
|
|
|
std::shared_ptr<i2p::tunnel::TunnelPool> m_Pool;
|
2016-10-25 21:07:34 +03:00
|
|
|
std::mutex m_LeaseSetMutex;
|
2016-05-25 22:10:28 +03:00
|
|
|
std::shared_ptr<i2p::data::LocalLeaseSet> m_LeaseSet;
|
2014-10-16 20:37:39 +04:00
|
|
|
bool m_IsPublic;
|
2014-11-28 21:01:35 +03:00
|
|
|
uint32_t m_PublishReplyToken;
|
2014-12-08 00:10:25 +03:00
|
|
|
std::set<i2p::data::IdentHash> m_ExcludedFloodfills; // for publishing
|
2014-10-22 19:46:54 +04:00
|
|
|
|
2016-02-11 22:45:33 +03:00
|
|
|
boost::asio::deadline_timer m_PublishConfirmationTimer, m_PublishVerificationTimer, m_CleanupTimer;
|
2014-11-28 21:01:35 +03:00
|
|
|
|
2014-10-16 20:37:39 +04:00
|
|
|
public:
|
2014-10-13 00:22:14 +04:00
|
|
|
|
2014-10-16 20:37:39 +04:00
|
|
|
// for HTTP only
|
|
|
|
int GetNumRemoteLeaseSets () const { return m_RemoteLeaseSets.size (); };
|
2016-11-12 17:49:16 +03:00
|
|
|
const decltype(m_RemoteLeaseSets)& GetLeaseSets () const { return m_RemoteLeaseSets; };
|
2014-10-16 20:37:39 +04:00
|
|
|
};
|
2016-05-23 17:33:01 +03:00
|
|
|
|
|
|
|
class ClientDestination: public LeaseSetDestination
|
|
|
|
{
|
|
|
|
public:
|
2016-10-19 17:23:02 +03:00
|
|
|
#ifdef I2LUA
|
2016-08-30 20:27:57 +03:00
|
|
|
// type for informing that a client destination is ready
|
|
|
|
typedef std::promise<std::shared_ptr<ClientDestination> > ReadyPromise;
|
2016-10-19 17:23:02 +03:00
|
|
|
// informs promise with shared_from_this() when this destination is ready to use
|
|
|
|
// if cancelled before ready, informs promise with nullptr
|
|
|
|
void Ready(ReadyPromise & p);
|
|
|
|
#endif
|
2016-08-30 20:27:57 +03:00
|
|
|
|
2016-05-23 17:33:01 +03:00
|
|
|
ClientDestination (const i2p::data::PrivateKeys& keys, bool isPublic, const std::map<std::string, std::string> * params = nullptr);
|
2016-05-23 21:31:22 +03:00
|
|
|
~ClientDestination ();
|
|
|
|
|
|
|
|
bool Start ();
|
|
|
|
bool Stop ();
|
2016-08-30 20:27:57 +03:00
|
|
|
|
2016-05-26 21:54:33 +03:00
|
|
|
const i2p::data::PrivateKeys& GetPrivateKeys () const { return m_Keys; };
|
2016-05-29 16:33:50 +03:00
|
|
|
void Sign (const uint8_t * buf, int len, uint8_t * signature) const { m_Keys.Sign (buf, len, signature); };
|
|
|
|
|
2016-05-23 21:31:22 +03:00
|
|
|
// streaming
|
|
|
|
std::shared_ptr<i2p::stream::StreamingDestination> CreateStreamingDestination (int port, bool gzip = true); // additional
|
|
|
|
std::shared_ptr<i2p::stream::StreamingDestination> GetStreamingDestination (int port = 0) const;
|
|
|
|
// following methods operate with default streaming destination
|
|
|
|
void CreateStream (StreamRequestComplete streamRequestComplete, const i2p::data::IdentHash& dest, int port = 0);
|
|
|
|
std::shared_ptr<i2p::stream::Stream> CreateStream (std::shared_ptr<const i2p::data::LeaseSet> remote, int port = 0);
|
|
|
|
void AcceptStreams (const i2p::stream::StreamingDestination::Acceptor& acceptor);
|
|
|
|
void StopAcceptingStreams ();
|
|
|
|
bool IsAcceptingStreams () const;
|
2016-05-29 16:33:50 +03:00
|
|
|
|
2016-05-23 21:31:22 +03:00
|
|
|
// datagram
|
2016-08-22 05:29:55 +03:00
|
|
|
i2p::datagram::DatagramDestination * GetDatagramDestination () const { return m_DatagramDestination; };
|
|
|
|
i2p::datagram::DatagramDestination * CreateDatagramDestination ();
|
2016-05-23 21:31:22 +03:00
|
|
|
|
2016-05-29 16:33:50 +03:00
|
|
|
// implements LocalDestination
|
|
|
|
const uint8_t * GetEncryptionPrivateKey () const { return m_EncryptionPrivateKey; };
|
|
|
|
std::shared_ptr<const i2p::data::IdentityEx> GetIdentity () const { return m_Keys.GetPublic (); };
|
2016-05-23 17:33:01 +03:00
|
|
|
|
2016-05-23 21:31:22 +03:00
|
|
|
protected:
|
|
|
|
|
2016-09-08 17:16:42 +03:00
|
|
|
void CleanupDestination ();
|
2016-05-23 21:31:22 +03:00
|
|
|
// I2CP
|
|
|
|
void HandleDataMessage (const uint8_t * buf, size_t len);
|
2016-05-29 16:33:50 +03:00
|
|
|
void CreateNewLeaseSet (std::vector<std::shared_ptr<i2p::tunnel::InboundTunnel> > tunnels);
|
2016-05-23 21:31:22 +03:00
|
|
|
|
2016-05-23 17:33:01 +03:00
|
|
|
private:
|
|
|
|
|
2016-05-25 23:18:02 +03:00
|
|
|
std::shared_ptr<ClientDestination> GetSharedFromThis ()
|
2016-05-29 16:33:50 +03:00
|
|
|
{ return std::static_pointer_cast<ClientDestination>(shared_from_this ()); }
|
|
|
|
void PersistTemporaryKeys ();
|
2016-10-19 17:23:02 +03:00
|
|
|
#ifdef I2LUA
|
2016-08-30 20:27:57 +03:00
|
|
|
void ScheduleCheckForReady(ReadyPromise * p);
|
|
|
|
void HandleCheckForReady(const boost::system::error_code & ecode, ReadyPromise * p);
|
2016-10-19 17:23:02 +03:00
|
|
|
#endif
|
2016-05-23 17:33:01 +03:00
|
|
|
private:
|
|
|
|
|
|
|
|
i2p::data::PrivateKeys m_Keys;
|
2016-05-29 16:33:50 +03:00
|
|
|
uint8_t m_EncryptionPublicKey[256], m_EncryptionPrivateKey[256];
|
2016-05-23 21:31:22 +03:00
|
|
|
|
|
|
|
std::shared_ptr<i2p::stream::StreamingDestination> m_StreamingDestination; // default
|
|
|
|
std::map<uint16_t, std::shared_ptr<i2p::stream::StreamingDestination> > m_StreamingDestinationsByPorts;
|
2016-08-22 05:29:55 +03:00
|
|
|
i2p::datagram::DatagramDestination * m_DatagramDestination;
|
2016-05-23 21:31:22 +03:00
|
|
|
|
2016-08-30 20:27:57 +03:00
|
|
|
boost::asio::deadline_timer m_ReadyChecker;
|
|
|
|
|
2016-05-23 21:31:22 +03:00
|
|
|
public:
|
|
|
|
|
|
|
|
// for HTTP only
|
|
|
|
std::vector<std::shared_ptr<const i2p::stream::Stream> > GetAllStreams () const;
|
2016-05-23 17:33:01 +03:00
|
|
|
};
|
2014-10-22 19:46:54 +04:00
|
|
|
}
|
2014-10-05 16:54:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|