mirror of https://github.com/PurpleI2P/i2pd.git
I2P: End-to-End encrypted and anonymous Internet
https://i2pd.website/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
247 lines
9.3 KiB
247 lines
9.3 KiB
/* |
|
* Copyright (c) 2013-2023, 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 |
|
*/ |
|
|
|
#ifndef ROUTER_CONTEXT_H__ |
|
#define ROUTER_CONTEXT_H__ |
|
|
|
#include <inttypes.h> |
|
#include <string> |
|
#include <memory> |
|
#include <chrono> |
|
#include <set> |
|
#include <boost/asio.hpp> |
|
#include "Identity.h" |
|
#include "RouterInfo.h" |
|
#include "Garlic.h" |
|
#include "util.h" |
|
|
|
namespace i2p |
|
{ |
|
namespace garlic |
|
{ |
|
class RouterIncomingRatchetSession; |
|
} |
|
|
|
const char ROUTER_INFO[] = "router.info"; |
|
const char ROUTER_KEYS[] = "router.keys"; |
|
const char NTCP2_KEYS[] = "ntcp2.keys"; |
|
const char SSU2_KEYS[] = "ssu2.keys"; |
|
const int ROUTER_INFO_UPDATE_INTERVAL = 30*60; // 30 minutes |
|
const int ROUTER_INFO_PUBLISH_INTERVAL = 39*60; // in seconds |
|
const int ROUTER_INFO_INITIAL_PUBLISH_INTERVAL = 10; // in seconds |
|
const int ROUTER_INFO_PUBLISH_INTERVAL_VARIANCE = 105;// in seconds |
|
const int ROUTER_INFO_CONFIRMATION_TIMEOUT = 5; // in seconds |
|
const int ROUTER_INFO_MAX_PUBLISH_EXCLUDED_FLOODFILLS = 15; |
|
|
|
enum RouterStatus |
|
{ |
|
eRouterStatusOK = 0, |
|
eRouterStatusTesting = 1, |
|
eRouterStatusFirewalled = 2, |
|
eRouterStatusUnknown = 3, |
|
eRouterStatusProxy = 4, |
|
eRouterStatusMesh = 5 |
|
}; |
|
|
|
enum RouterError |
|
{ |
|
eRouterErrorNone = 0, |
|
eRouterErrorClockSkew = 1, |
|
eRouterErrorOffline = 2, |
|
eRouterErrorSymmetricNAT = 3, |
|
eRouterErrorFullConeNAT = 4, |
|
eRouterErrorNoDescriptors = 5 |
|
}; |
|
|
|
class RouterContext: public i2p::garlic::GarlicDestination |
|
{ |
|
private: |
|
|
|
struct NTCP2PrivateKeys |
|
{ |
|
uint8_t staticPublicKey[32]; |
|
uint8_t staticPrivateKey[32]; |
|
uint8_t iv[16]; |
|
}; |
|
|
|
struct SSU2PrivateKeys |
|
{ |
|
uint8_t staticPublicKey[32]; |
|
uint8_t staticPrivateKey[32]; |
|
uint8_t intro[32]; |
|
}; |
|
|
|
class RouterService: public i2p::util::RunnableServiceWithWork |
|
{ |
|
public: |
|
|
|
RouterService (): RunnableServiceWithWork ("Router") {}; |
|
boost::asio::io_service& GetService () { return GetIOService (); }; |
|
void Start () { StartIOService (); }; |
|
void Stop () { StopIOService (); }; |
|
}; |
|
|
|
public: |
|
|
|
RouterContext (); |
|
void Init (); |
|
void Start (); |
|
void Stop (); |
|
|
|
const i2p::data::PrivateKeys& GetPrivateKeys () const { return m_Keys; }; |
|
i2p::data::LocalRouterInfo& GetRouterInfo () { return m_RouterInfo; }; |
|
std::shared_ptr<i2p::data::RouterInfo> GetSharedRouterInfo () |
|
{ |
|
return std::shared_ptr<i2p::data::RouterInfo> (&m_RouterInfo, |
|
[](i2p::data::RouterInfo *) {}); |
|
} |
|
std::shared_ptr<i2p::garlic::GarlicDestination> GetSharedDestination () |
|
{ |
|
return std::shared_ptr<i2p::garlic::GarlicDestination> (this, |
|
[](i2p::garlic::GarlicDestination *) {}); |
|
} |
|
|
|
const uint8_t * GetNTCP2StaticPublicKey () const { return m_NTCP2Keys ? m_NTCP2Keys->staticPublicKey : nullptr; }; |
|
const uint8_t * GetNTCP2StaticPrivateKey () const { return m_NTCP2Keys ? m_NTCP2Keys->staticPrivateKey : nullptr; }; |
|
const uint8_t * GetNTCP2IV () const { return m_NTCP2Keys ? m_NTCP2Keys->iv : nullptr; }; |
|
i2p::crypto::X25519Keys& GetNTCP2StaticKeys (); |
|
|
|
const uint8_t * GetSSU2StaticPublicKey () const { return m_SSU2Keys ? m_SSU2Keys->staticPublicKey : nullptr; }; |
|
const uint8_t * GetSSU2StaticPrivateKey () const { return m_SSU2Keys ? m_SSU2Keys->staticPrivateKey : nullptr; }; |
|
const uint8_t * GetSSU2IntroKey () const { return m_SSU2Keys ? m_SSU2Keys->intro : nullptr; }; |
|
i2p::crypto::X25519Keys& GetSSU2StaticKeys (); |
|
|
|
uint32_t GetUptime () const; // in seconds |
|
uint64_t GetLastUpdateTime () const { return m_LastUpdateTime; }; |
|
uint64_t GetBandwidthLimit () const { return m_BandwidthLimit; }; |
|
uint64_t GetTransitBandwidthLimit () const { return (m_BandwidthLimit*m_ShareRatio)/100LL; }; |
|
RouterStatus GetStatus () const { return m_Status; }; |
|
void SetStatus (RouterStatus status); |
|
RouterError GetError () const { return m_Error; }; |
|
void SetError (RouterError error) { m_Error = error; }; |
|
RouterStatus GetStatusV6 () const { return m_StatusV6; }; |
|
void SetStatusV6 (RouterStatus status); |
|
RouterError GetErrorV6 () const { return m_ErrorV6; }; |
|
void SetErrorV6 (RouterError error) { m_ErrorV6 = error; }; |
|
int GetNetID () const { return m_NetID; }; |
|
void SetNetID (int netID) { m_NetID = netID; }; |
|
bool DecryptTunnelBuildRecord (const uint8_t * encrypted, uint8_t * data); |
|
bool DecryptTunnelShortRequestRecord (const uint8_t * encrypted, uint8_t * data); |
|
|
|
void UpdatePort (int port); // called from Daemon |
|
void UpdateAddress (const boost::asio::ip::address& host); // called from SSU2 or Daemon |
|
void PublishNTCP2Address (int port, bool publish, bool v4, bool v6, bool ygg); |
|
void PublishSSU2Address (int port, bool publish, bool v4, bool v6); |
|
bool AddSSU2Introducer (const i2p::data::RouterInfo::Introducer& introducer, bool v4); |
|
void RemoveSSU2Introducer (const i2p::data::IdentHash& h, bool v4); |
|
void ClearSSU2Introducers (bool v4); |
|
bool IsUnreachable () const; |
|
void SetUnreachable (bool v4, bool v6); |
|
void SetReachable (bool v4, bool v6); |
|
bool IsFloodfill () const { return m_IsFloodfill; }; |
|
void SetFloodfill (bool floodfill); |
|
void SetFamily (const std::string& family); |
|
std::string GetFamily () const; |
|
void SetBandwidth (int limit); /* in kilobytes */ |
|
void SetBandwidth (char L); /* by letter */ |
|
void SetShareRatio (int percents); // 0 - 100 |
|
bool AcceptsTunnels () const { return m_AcceptsTunnels; }; |
|
void SetAcceptsTunnels (bool acceptsTunnels) { m_AcceptsTunnels = acceptsTunnels; }; |
|
bool SupportsV6 () const { return m_RouterInfo.IsV6 (); }; |
|
bool SupportsV4 () const { return m_RouterInfo.IsV4 (); }; |
|
bool SupportsMesh () const { return m_RouterInfo.IsMesh (); }; |
|
void SetSupportsV6 (bool supportsV6); |
|
void SetSupportsV4 (bool supportsV4); |
|
void SetSupportsMesh (bool supportsmesh, const boost::asio::ip::address_v6& host); |
|
void SetMTU (int mtu, bool v4); |
|
void SetHidden(bool hide) { m_IsHiddenMode = hide; }; |
|
bool IsHidden() const { return m_IsHiddenMode; }; |
|
i2p::crypto::NoiseSymmetricState& GetCurrentNoiseState () { return m_CurrentNoiseState; }; |
|
|
|
void UpdateNTCP2V6Address (const boost::asio::ip::address& host); // called from Daemon. TODO: remove |
|
void UpdateStats (); |
|
void UpdateTimestamp (uint64_t ts); // in seconds, called from NetDb before publishing |
|
void CleanupDestination (); // garlic destination |
|
|
|
// implements LocalDestination |
|
std::shared_ptr<const i2p::data::IdentityEx> GetIdentity () const { return m_Keys.GetPublic (); }; |
|
bool Decrypt (const uint8_t * encrypted, uint8_t * data, i2p::data::CryptoKeyType preferredCrypto) const; |
|
void Sign (const uint8_t * buf, int len, uint8_t * signature) const { m_Keys.Sign (buf, len, signature); }; |
|
void SetLeaseSetUpdated () {}; |
|
|
|
// implements GarlicDestination |
|
std::shared_ptr<const i2p::data::LocalLeaseSet> GetLeaseSet () { return nullptr; }; |
|
std::shared_ptr<i2p::tunnel::TunnelPool> GetTunnelPool () const; |
|
|
|
// override GarlicDestination |
|
void ProcessGarlicMessage (std::shared_ptr<I2NPMessage> msg); |
|
void ProcessDeliveryStatusMessage (std::shared_ptr<I2NPMessage> msg); |
|
|
|
protected: |
|
|
|
// implements GarlicDestination |
|
void HandleI2NPMessage (const uint8_t * buf, size_t len); |
|
bool HandleCloveI2NPMessage (I2NPMessageType typeID, const uint8_t * payload, size_t len, uint32_t msgID); |
|
|
|
private: |
|
|
|
void CreateNewRouter (); |
|
void NewRouterInfo (); |
|
void UpdateRouterInfo (); |
|
void NewNTCP2Keys (); |
|
void NewSSU2Keys (); |
|
void UpdateNTCP2Keys (); |
|
void UpdateSSU2Keys (); |
|
bool Load (); |
|
void SaveKeys (); |
|
uint16_t SelectRandomPort () const; |
|
void PublishNTCP2Address (std::shared_ptr<i2p::data::RouterInfo::Address> address, int port, bool publish) const; |
|
|
|
bool DecryptECIESTunnelBuildRecord (const uint8_t * encrypted, uint8_t * data, size_t clearTextSize); |
|
void PostGarlicMessage (std::shared_ptr<I2NPMessage> msg); |
|
void PostDeliveryStatusMessage (std::shared_ptr<I2NPMessage> msg); |
|
|
|
void ScheduleInitialPublish (); |
|
void HandleInitialPublishTimer (const boost::system::error_code& ecode); |
|
void SchedulePublish (); |
|
void HandlePublishTimer (const boost::system::error_code& ecode); |
|
void Publish (); |
|
void SchedulePublishResend (); |
|
void HandlePublishResendTimer (const boost::system::error_code& ecode); |
|
|
|
private: |
|
|
|
i2p::data::LocalRouterInfo m_RouterInfo; |
|
i2p::data::PrivateKeys m_Keys; |
|
std::shared_ptr<i2p::crypto::CryptoKeyDecryptor> m_Decryptor, m_TunnelDecryptor; |
|
std::shared_ptr<i2p::garlic::RouterIncomingRatchetSession> m_ECIESSession; |
|
uint64_t m_LastUpdateTime; // in seconds |
|
bool m_AcceptsTunnels, m_IsFloodfill; |
|
std::chrono::time_point<std::chrono::steady_clock> m_StartupTime; |
|
uint64_t m_BandwidthLimit; // allowed bandwidth |
|
int m_ShareRatio; |
|
RouterStatus m_Status, m_StatusV6; |
|
RouterError m_Error, m_ErrorV6; |
|
int m_NetID; |
|
std::unique_ptr<NTCP2PrivateKeys> m_NTCP2Keys; |
|
std::unique_ptr<SSU2PrivateKeys> m_SSU2Keys; |
|
std::unique_ptr<i2p::crypto::X25519Keys> m_NTCP2StaticKeys, m_SSU2StaticKeys; |
|
// for ECIESx25519 |
|
i2p::crypto::NoiseSymmetricState m_InitialNoiseState, m_CurrentNoiseState; |
|
// publish |
|
std::unique_ptr<RouterService> m_Service; |
|
std::unique_ptr<boost::asio::deadline_timer> m_PublishTimer; |
|
std::set<i2p::data::IdentHash> m_PublishExcluded; |
|
uint32_t m_PublishReplyToken; |
|
bool m_IsHiddenMode; // not publish |
|
}; |
|
|
|
extern RouterContext context; |
|
} |
|
|
|
#endif
|
|
|