2020-05-22 16:18:41 +03:00
|
|
|
/*
|
2024-02-17 02:56:04 +03:00
|
|
|
* Copyright (c) 2013-2024, The PurpleI2P Project
|
2020-05-22 16:18:41 +03: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
|
|
|
|
*/
|
|
|
|
|
2013-10-23 06:45:40 +04:00
|
|
|
#ifndef ROUTER_CONTEXT_H__
|
|
|
|
#define ROUTER_CONTEXT_H__
|
|
|
|
|
|
|
|
#include <inttypes.h>
|
2014-10-27 22:08:50 +03:00
|
|
|
#include <string>
|
2014-11-20 23:48:28 +03:00
|
|
|
#include <memory>
|
2024-05-15 20:31:31 +03:00
|
|
|
#include <unordered_set>
|
2014-10-29 20:49:21 +03:00
|
|
|
#include <boost/asio.hpp>
|
2014-04-01 21:42:04 +04:00
|
|
|
#include "Identity.h"
|
2013-10-23 06:45:40 +04:00
|
|
|
#include "RouterInfo.h"
|
2014-10-07 04:18:18 +04:00
|
|
|
#include "Garlic.h"
|
2023-02-23 21:58:06 +03:00
|
|
|
#include "util.h"
|
2013-10-23 06:45:40 +04:00
|
|
|
|
|
|
|
namespace i2p
|
|
|
|
{
|
2021-03-03 23:30:13 +03:00
|
|
|
namespace garlic
|
|
|
|
{
|
|
|
|
class RouterIncomingRatchetSession;
|
2021-05-22 18:29:05 +03:00
|
|
|
}
|
2021-03-03 23:30:13 +03:00
|
|
|
|
2013-10-23 06:45:40 +04:00
|
|
|
const char ROUTER_INFO[] = "router.info";
|
2018-01-06 06:48:51 +03:00
|
|
|
const char ROUTER_KEYS[] = "router.keys";
|
2020-03-01 13:25:50 +03:00
|
|
|
const char NTCP2_KEYS[] = "ntcp2.keys";
|
2022-03-02 05:23:08 +03:00
|
|
|
const char SSU2_KEYS[] = "ssu2.keys";
|
2023-02-25 02:28:30 +03:00
|
|
|
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;
|
2023-03-07 03:48:04 +03:00
|
|
|
const int ROUTER_INFO_CONGESTION_UPDATE_INTERVAL = 12*60; // in seconds
|
2024-02-17 02:56:04 +03:00
|
|
|
const int ROUTER_INFO_CLEANUP_INTERVAL = 5; // in minutes
|
2015-03-18 22:06:15 +03:00
|
|
|
|
2015-02-26 21:44:18 +03:00
|
|
|
enum RouterStatus
|
|
|
|
{
|
|
|
|
eRouterStatusOK = 0,
|
2023-07-30 15:29:10 +03:00
|
|
|
eRouterStatusFirewalled = 1,
|
|
|
|
eRouterStatusUnknown = 2,
|
|
|
|
eRouterStatusProxy = 3,
|
|
|
|
eRouterStatusMesh = 4
|
2018-01-06 06:48:51 +03:00
|
|
|
};
|
2015-02-26 21:44:18 +03:00
|
|
|
|
2024-01-28 16:37:14 +03:00
|
|
|
const char* const ROUTER_STATUS_NAMES[] =
|
|
|
|
{
|
|
|
|
"OK", // 0
|
|
|
|
"Firewalled", // 1
|
|
|
|
"Unknown", // 2
|
|
|
|
"Proxy", // 3
|
|
|
|
"Mesh" // 4
|
|
|
|
};
|
|
|
|
|
2016-09-20 04:37:04 +03:00
|
|
|
enum RouterError
|
|
|
|
{
|
|
|
|
eRouterErrorNone = 0,
|
2020-10-12 00:51:40 +03:00
|
|
|
eRouterErrorClockSkew = 1,
|
2021-03-01 20:20:53 +03:00
|
|
|
eRouterErrorOffline = 2,
|
2022-12-10 21:22:37 +03:00
|
|
|
eRouterErrorSymmetricNAT = 3,
|
2023-01-07 20:11:51 +03:00
|
|
|
eRouterErrorFullConeNAT = 4,
|
|
|
|
eRouterErrorNoDescriptors = 5
|
2018-01-06 06:48:51 +03:00
|
|
|
};
|
2021-05-22 18:29:05 +03:00
|
|
|
|
2023-02-26 16:46:01 +03:00
|
|
|
class RouterContext: public i2p::garlic::GarlicDestination
|
2013-10-23 06:45:40 +04:00
|
|
|
{
|
2018-06-11 22:33:48 +03:00
|
|
|
private:
|
|
|
|
|
2020-03-01 13:25:50 +03:00
|
|
|
struct NTCP2PrivateKeys
|
2018-06-11 22:33:48 +03:00
|
|
|
{
|
2018-06-15 19:52:43 +03:00
|
|
|
uint8_t staticPublicKey[32];
|
|
|
|
uint8_t staticPrivateKey[32];
|
2018-06-11 22:33:48 +03:00
|
|
|
uint8_t iv[16];
|
2020-03-01 13:25:50 +03:00
|
|
|
};
|
2018-06-11 22:33:48 +03:00
|
|
|
|
2022-03-02 05:23:08 +03:00
|
|
|
struct SSU2PrivateKeys
|
|
|
|
{
|
|
|
|
uint8_t staticPublicKey[32];
|
|
|
|
uint8_t staticPrivateKey[32];
|
|
|
|
uint8_t intro[32];
|
|
|
|
};
|
2022-05-20 19:56:05 +03:00
|
|
|
|
2023-02-26 16:46:01 +03:00
|
|
|
class RouterService: public i2p::util::RunnableServiceWithWork
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
RouterService (): RunnableServiceWithWork ("Router") {};
|
|
|
|
boost::asio::io_service& GetService () { return GetIOService (); };
|
|
|
|
void Start () { StartIOService (); };
|
|
|
|
void Stop () { StopIOService (); };
|
|
|
|
};
|
|
|
|
|
2013-10-23 06:45:40 +04:00
|
|
|
public:
|
|
|
|
|
|
|
|
RouterContext ();
|
2014-09-04 17:31:42 +04:00
|
|
|
void Init ();
|
2023-02-23 21:58:06 +03:00
|
|
|
void Start ();
|
|
|
|
void Stop ();
|
|
|
|
|
2016-05-26 21:54:33 +03:00
|
|
|
const i2p::data::PrivateKeys& GetPrivateKeys () const { return m_Keys; };
|
2022-01-16 02:54:02 +03:00
|
|
|
i2p::data::LocalRouterInfo& GetRouterInfo () { return m_RouterInfo; };
|
2021-07-14 21:46:56 +03:00
|
|
|
std::shared_ptr<i2p::data::RouterInfo> GetSharedRouterInfo ()
|
2018-01-06 06:48:51 +03:00
|
|
|
{
|
2021-07-14 21:46:56 +03:00
|
|
|
return std::shared_ptr<i2p::data::RouterInfo> (&m_RouterInfo,
|
|
|
|
[](i2p::data::RouterInfo *) {});
|
2014-11-20 23:48:28 +03:00
|
|
|
}
|
2018-01-06 06:48:51 +03:00
|
|
|
std::shared_ptr<i2p::garlic::GarlicDestination> GetSharedDestination ()
|
2015-12-16 22:52:48 +03:00
|
|
|
{
|
2018-01-06 06:48:51 +03:00
|
|
|
return std::shared_ptr<i2p::garlic::GarlicDestination> (this,
|
2015-12-16 22:52:48 +03:00
|
|
|
[](i2p::garlic::GarlicDestination *) {});
|
2018-01-06 06:48:51 +03:00
|
|
|
}
|
2024-05-26 22:33:37 +03:00
|
|
|
std::shared_ptr<i2p::data::RouterInfo::Buffer> CopyRouterInfoBuffer () const;
|
|
|
|
|
2018-06-15 19:52:43 +03:00
|
|
|
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; };
|
2022-03-02 05:23:08 +03:00
|
|
|
i2p::crypto::X25519Keys& GetNTCP2StaticKeys ();
|
2018-01-06 06:48:51 +03:00
|
|
|
|
2022-03-02 05:23:08 +03:00
|
|
|
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 ();
|
2022-05-20 19:56:05 +03:00
|
|
|
|
2019-06-19 18:43:04 +03:00
|
|
|
uint32_t GetUptime () const; // in seconds
|
2015-03-18 22:06:15 +03:00
|
|
|
uint64_t GetLastUpdateTime () const { return m_LastUpdateTime; };
|
2016-03-31 03:00:00 +03:00
|
|
|
uint64_t GetBandwidthLimit () const { return m_BandwidthLimit; };
|
2017-05-04 21:58:12 +03:00
|
|
|
uint64_t GetTransitBandwidthLimit () const { return (m_BandwidthLimit*m_ShareRatio)/100LL; };
|
2023-07-30 15:29:10 +03:00
|
|
|
bool GetTesting () const { return m_Testing; };
|
|
|
|
void SetTesting (bool testing);
|
2015-02-26 21:44:18 +03:00
|
|
|
RouterStatus GetStatus () const { return m_Status; };
|
2015-11-03 17:15:49 +03:00
|
|
|
void SetStatus (RouterStatus status);
|
2016-09-20 04:37:04 +03:00
|
|
|
RouterError GetError () const { return m_Error; };
|
2022-08-30 04:11:17 +03:00
|
|
|
void SetError (RouterError error) { m_Error = error; };
|
2023-07-30 15:29:10 +03:00
|
|
|
bool GetTestingV6 () const { return m_TestingV6; };
|
|
|
|
void SetTestingV6 (bool testing);
|
2021-03-23 22:36:57 +03:00
|
|
|
RouterStatus GetStatusV6 () const { return m_StatusV6; };
|
|
|
|
void SetStatusV6 (RouterStatus status);
|
2022-08-03 03:02:55 +03:00
|
|
|
RouterError GetErrorV6 () const { return m_ErrorV6; };
|
2022-08-30 04:11:17 +03:00
|
|
|
void SetErrorV6 (RouterError error) { m_ErrorV6 = error; };
|
2016-10-12 18:26:48 +03:00
|
|
|
int GetNetID () const { return m_NetID; };
|
2018-01-06 06:48:51 +03:00
|
|
|
void SetNetID (int netID) { m_NetID = netID; };
|
2021-03-28 19:14:02 +03:00
|
|
|
bool DecryptTunnelBuildRecord (const uint8_t * encrypted, uint8_t * data);
|
2021-06-03 02:50:29 +03:00
|
|
|
bool DecryptTunnelShortRequestRecord (const uint8_t * encrypted, uint8_t * data);
|
2021-11-27 23:30:35 +03:00
|
|
|
|
2020-03-01 13:25:50 +03:00
|
|
|
void UpdatePort (int port); // called from Daemon
|
2023-01-25 21:23:46 +03:00
|
|
|
void UpdateAddress (const boost::asio::ip::address& host); // called from SSU2 or Daemon
|
2021-03-16 20:08:10 +03:00
|
|
|
void PublishNTCP2Address (int port, bool publish, bool v4, bool v6, bool ygg);
|
2022-03-29 20:56:56 +03:00
|
|
|
void PublishSSU2Address (int port, bool publish, bool v4, bool v6);
|
2022-07-20 23:01:08 +03:00
|
|
|
bool AddSSU2Introducer (const i2p::data::RouterInfo::Introducer& introducer, bool v4);
|
|
|
|
void RemoveSSU2Introducer (const i2p::data::IdentHash& h, bool v4);
|
2024-09-27 20:32:20 +03:00
|
|
|
void UpdateSSU2Introducer (const i2p::data::IdentHash& h, bool v4, uint32_t iTag, uint32_t iExp);
|
2022-08-06 04:23:23 +03:00
|
|
|
void ClearSSU2Introducers (bool v4);
|
2015-03-01 15:55:03 +03:00
|
|
|
bool IsUnreachable () const;
|
2021-03-27 22:16:56 +03:00
|
|
|
void SetUnreachable (bool v4, bool v6);
|
|
|
|
void SetReachable (bool v4, bool v6);
|
2018-01-06 06:48:51 +03:00
|
|
|
bool IsFloodfill () const { return m_IsFloodfill; };
|
|
|
|
void SetFloodfill (bool floodfill);
|
2016-02-21 04:20:19 +03:00
|
|
|
void SetFamily (const std::string& family);
|
2016-04-29 01:16:11 +03:00
|
|
|
std::string GetFamily () const;
|
2016-03-31 03:00:00 +03:00
|
|
|
void SetBandwidth (int limit); /* in kilobytes */
|
|
|
|
void SetBandwidth (char L); /* by letter */
|
2017-05-04 21:58:12 +03:00
|
|
|
void SetShareRatio (int percents); // 0 - 100
|
2014-09-30 21:34:29 +04:00
|
|
|
bool AcceptsTunnels () const { return m_AcceptsTunnels; };
|
|
|
|
void SetAcceptsTunnels (bool acceptsTunnels) { m_AcceptsTunnels = acceptsTunnels; };
|
2024-02-18 23:54:43 +03:00
|
|
|
int GetCongestionLevel (bool longTerm) const;
|
2014-10-27 04:32:06 +03:00
|
|
|
bool SupportsV6 () const { return m_RouterInfo.IsV6 (); };
|
2016-03-25 01:44:41 +03:00
|
|
|
bool SupportsV4 () const { return m_RouterInfo.IsV4 (); };
|
2021-02-01 01:25:07 +03:00
|
|
|
bool SupportsMesh () const { return m_RouterInfo.IsMesh (); };
|
2014-10-27 04:32:06 +03:00
|
|
|
void SetSupportsV6 (bool supportsV6);
|
2016-03-25 01:44:41 +03:00
|
|
|
void SetSupportsV4 (bool supportsV4);
|
2021-02-01 02:30:53 +03:00
|
|
|
void SetSupportsMesh (bool supportsmesh, const boost::asio::ip::address_v6& host);
|
2022-07-24 22:24:01 +03:00
|
|
|
void SetMTU (int mtu, bool v4);
|
2023-02-25 02:28:30 +03:00
|
|
|
void SetHidden(bool hide) { m_IsHiddenMode = hide; };
|
|
|
|
bool IsHidden() const { return m_IsHiddenMode; };
|
2024-09-07 23:25:26 +03:00
|
|
|
bool IsLimitedConnectivity () const { return m_Status == eRouterStatusProxy; }; // TODO: implement other cases
|
2021-06-09 19:49:50 +03:00
|
|
|
i2p::crypto::NoiseSymmetricState& GetCurrentNoiseState () { return m_CurrentNoiseState; };
|
2021-05-25 22:03:29 +03:00
|
|
|
|
2019-05-29 18:36:58 +03:00
|
|
|
void UpdateNTCP2V6Address (const boost::asio::ip::address& host); // called from Daemon. TODO: remove
|
2018-01-06 06:48:51 +03:00
|
|
|
void UpdateStats ();
|
2018-09-21 17:13:18 +03:00
|
|
|
void UpdateTimestamp (uint64_t ts); // in seconds, called from NetDb before publishing
|
2014-10-27 22:08:50 +03:00
|
|
|
|
2014-04-01 21:42:04 +04:00
|
|
|
// implements LocalDestination
|
2024-10-19 02:57:35 +03:00
|
|
|
std::shared_ptr<const i2p::data::IdentityEx> GetIdentity () const override{ return m_Keys.GetPublic (); };
|
|
|
|
bool Decrypt (const uint8_t * encrypted, uint8_t * data, i2p::data::CryptoKeyType preferredCrypto) const override;
|
|
|
|
void SetLeaseSetUpdated (bool post) override {};
|
2014-10-08 05:08:00 +04:00
|
|
|
|
|
|
|
// implements GarlicDestination
|
2024-10-19 02:57:35 +03:00
|
|
|
std::shared_ptr<const i2p::data::LocalLeaseSet> GetLeaseSet () override { return nullptr; };
|
|
|
|
std::shared_ptr<i2p::tunnel::TunnelPool> GetTunnelPool () const override;
|
2015-06-10 05:14:31 +03:00
|
|
|
|
|
|
|
// override GarlicDestination
|
2024-10-19 02:57:35 +03:00
|
|
|
void ProcessGarlicMessage (std::shared_ptr<I2NPMessage> msg) override;
|
|
|
|
void ProcessDeliveryStatusMessage (std::shared_ptr<I2NPMessage> msg) override;
|
2024-10-19 03:02:41 +03:00
|
|
|
void SubmitECIESx25519Key (const uint8_t * key, uint64_t tag) override;
|
2018-01-06 06:48:51 +03:00
|
|
|
|
2020-01-07 00:14:41 +03:00
|
|
|
protected:
|
|
|
|
|
|
|
|
// implements GarlicDestination
|
2024-10-19 02:57:35 +03:00
|
|
|
void HandleI2NPMessage (const uint8_t * buf, size_t len) override;
|
|
|
|
bool HandleCloveI2NPMessage (I2NPMessageType typeID, const uint8_t * payload, size_t len, uint32_t msgID) override;
|
2020-01-07 00:14:41 +03:00
|
|
|
|
2013-10-23 06:45:40 +04:00
|
|
|
private:
|
|
|
|
|
|
|
|
void CreateNewRouter ();
|
2014-09-01 00:46:39 +04:00
|
|
|
void NewRouterInfo ();
|
2014-02-23 20:48:09 +04:00
|
|
|
void UpdateRouterInfo ();
|
2018-06-11 22:33:48 +03:00
|
|
|
void NewNTCP2Keys ();
|
2022-03-02 05:23:08 +03:00
|
|
|
void NewSSU2Keys ();
|
2023-01-25 21:23:46 +03:00
|
|
|
void UpdateNTCP2Keys ();
|
|
|
|
void UpdateSSU2Keys ();
|
2013-10-23 06:45:40 +04:00
|
|
|
bool Load ();
|
2014-09-01 00:46:39 +04:00
|
|
|
void SaveKeys ();
|
2024-10-19 02:57:35 +03:00
|
|
|
void Sign (const uint8_t * buf, int len, uint8_t * signature) const { m_Keys.Sign (buf, len, signature); };
|
2022-08-11 05:00:11 +03:00
|
|
|
uint16_t SelectRandomPort () const;
|
2023-01-24 22:07:22 +03:00
|
|
|
void PublishNTCP2Address (std::shared_ptr<i2p::data::RouterInfo::Address> address, int port, bool publish) const;
|
2023-02-11 09:41:51 +03:00
|
|
|
|
2021-06-03 02:50:29 +03:00
|
|
|
bool DecryptECIESTunnelBuildRecord (const uint8_t * encrypted, uint8_t * data, size_t clearTextSize);
|
2023-02-23 21:58:06 +03:00
|
|
|
void PostGarlicMessage (std::shared_ptr<I2NPMessage> msg);
|
2023-02-25 02:28:30 +03:00
|
|
|
void PostDeliveryStatusMessage (std::shared_ptr<I2NPMessage> msg);
|
|
|
|
|
|
|
|
void ScheduleInitialPublish ();
|
2023-02-26 01:19:14 +03:00
|
|
|
void HandleInitialPublishTimer (const boost::system::error_code& ecode);
|
2023-02-25 02:28:30 +03:00
|
|
|
void SchedulePublish ();
|
|
|
|
void HandlePublishTimer (const boost::system::error_code& ecode);
|
|
|
|
void Publish ();
|
|
|
|
void SchedulePublishResend ();
|
|
|
|
void HandlePublishResendTimer (const boost::system::error_code& ecode);
|
2023-03-07 03:48:04 +03:00
|
|
|
void ScheduleCongestionUpdate ();
|
|
|
|
void HandleCongestionUpdateTimer (const boost::system::error_code& ecode);
|
2024-06-09 22:25:19 +03:00
|
|
|
void UpdateCongestion ();
|
2024-02-17 02:56:04 +03:00
|
|
|
void ScheduleCleanupTimer ();
|
|
|
|
void HandleCleanupTimer (const boost::system::error_code& ecode);
|
2023-02-23 21:58:06 +03:00
|
|
|
|
2013-10-23 06:45:40 +04:00
|
|
|
private:
|
|
|
|
|
2022-01-15 20:48:49 +03:00
|
|
|
i2p::data::LocalRouterInfo m_RouterInfo;
|
2018-01-06 06:48:51 +03:00
|
|
|
i2p::data::PrivateKeys m_Keys;
|
2021-02-27 05:02:51 +03:00
|
|
|
std::shared_ptr<i2p::crypto::CryptoKeyDecryptor> m_Decryptor, m_TunnelDecryptor;
|
2021-03-03 23:30:13 +03:00
|
|
|
std::shared_ptr<i2p::garlic::RouterIncomingRatchetSession> m_ECIESSession;
|
2018-09-21 17:13:18 +03:00
|
|
|
uint64_t m_LastUpdateTime; // in seconds
|
2020-03-01 13:25:50 +03:00
|
|
|
bool m_AcceptsTunnels, m_IsFloodfill;
|
2024-02-27 20:33:07 +03:00
|
|
|
uint64_t m_StartupTime; // monotonic seconds
|
2017-05-04 21:58:12 +03:00
|
|
|
uint64_t m_BandwidthLimit; // allowed bandwidth
|
|
|
|
int m_ShareRatio;
|
2021-03-23 22:36:57 +03:00
|
|
|
RouterStatus m_Status, m_StatusV6;
|
2022-08-03 03:02:55 +03:00
|
|
|
RouterError m_Error, m_ErrorV6;
|
2023-07-30 15:29:10 +03:00
|
|
|
bool m_Testing, m_TestingV6;
|
2016-10-12 18:26:48 +03:00
|
|
|
int m_NetID;
|
2018-06-11 22:33:48 +03:00
|
|
|
std::unique_ptr<NTCP2PrivateKeys> m_NTCP2Keys;
|
2022-03-02 05:23:08 +03:00
|
|
|
std::unique_ptr<SSU2PrivateKeys> m_SSU2Keys;
|
|
|
|
std::unique_ptr<i2p::crypto::X25519Keys> m_NTCP2StaticKeys, m_SSU2StaticKeys;
|
2020-11-03 02:49:07 +03:00
|
|
|
// for ECIESx25519
|
2021-06-09 19:49:50 +03:00
|
|
|
i2p::crypto::NoiseSymmetricState m_InitialNoiseState, m_CurrentNoiseState;
|
2023-02-25 02:28:30 +03:00
|
|
|
// publish
|
2023-02-26 16:46:01 +03:00
|
|
|
std::unique_ptr<RouterService> m_Service;
|
2024-02-17 02:56:04 +03:00
|
|
|
std::unique_ptr<boost::asio::deadline_timer> m_PublishTimer, m_CongestionUpdateTimer, m_CleanupTimer;
|
2024-05-15 20:31:31 +03:00
|
|
|
std::unordered_set<i2p::data::IdentHash> m_PublishExcluded;
|
2023-02-25 02:28:30 +03:00
|
|
|
uint32_t m_PublishReplyToken;
|
|
|
|
bool m_IsHiddenMode; // not publish
|
2024-05-26 22:33:37 +03:00
|
|
|
mutable std::mutex m_RouterInfoMutex;
|
2013-10-23 06:45:40 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
extern RouterContext context;
|
2018-01-06 06:48:51 +03:00
|
|
|
}
|
2013-10-23 06:45:40 +04:00
|
|
|
|
|
|
|
#endif
|