2022-06-05 19:33:36 -04:00
/*
2025-01-03 22:04:09 -05:00
* Copyright ( c ) 2022 - 2025 , The PurpleI2P Project
2022-06-05 19:33:36 -04: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
*/
# ifndef SSU2_SESSION_H__
# define SSU2_SESSION_H__
# include <memory>
# include <functional>
# include <map>
# include <set>
# include <list>
# include <boost/asio.hpp>
2024-10-26 15:30:48 -04:00
# include "version.h"
2022-06-05 19:33:36 -04:00
# include "Crypto.h"
# include "RouterInfo.h"
2022-07-06 12:41:51 -04:00
# include "RouterContext.h"
2022-06-05 19:33:36 -04:00
# include "TransportSession.h"
namespace i2p
{
namespace transport
{
const int SSU2_CONNECT_TIMEOUT = 5 ; // 5 seconds
2024-02-23 19:22:49 -05:00
const int SSU2_TERMINATION_TIMEOUT = 165 ; // in seconds
2022-07-13 12:45:20 -04:00
const int SSU2_CLOCK_SKEW = 60 ; // in seconds
2022-08-08 19:57:48 -04:00
const int SSU2_CLOCK_THRESHOLD = 15 ; // in seconds, if more we should adjust
2022-06-16 22:37:33 -04:00
const int SSU2_TOKEN_EXPIRATION_TIMEOUT = 9 ; // for Retry message, in seconds
const int SSU2_NEXT_TOKEN_EXPIRATION_TIMEOUT = 52 * 60 ; // for next token block, in seconds
2022-06-19 08:52:47 -04:00
const int SSU2_TOKEN_EXPIRATION_THRESHOLD = 2 ; // in seconds
2022-06-05 19:33:36 -04:00
const int SSU2_RELAY_NONCE_EXPIRATION_TIMEOUT = 10 ; // in seconds
const int SSU2_PEER_TEST_EXPIRATION_TIMEOUT = 60 ; // 60 seconds
2022-07-09 17:05:23 -04:00
const size_t SSU2_MAX_PACKET_SIZE = 1500 ;
2022-07-14 20:12:27 -04:00
const size_t SSU2_MIN_PACKET_SIZE = 1280 ;
2022-11-25 22:37:52 +02:00
const int SSU2_HANDSHAKE_RESEND_INTERVAL = 1000 ; // in milliseconds
2022-06-05 19:33:36 -04:00
const int SSU2_MAX_NUM_RESENDS = 5 ;
2024-08-14 10:13:35 -04:00
const int SSU2_RESEND_ATTEMPT_MIN_INTERVAL = 3 ; // in milliseconds
2022-06-05 19:33:36 -04:00
const int SSU2_INCOMPLETE_MESSAGES_CLEANUP_TIMEOUT = 30 ; // in seconds
2023-01-05 15:33:41 -05:00
const int SSU2_MAX_NUM_RECEIVED_I2NP_MSGIDS = 5000 ; // how many msgID we store for duplicates check
const int SSU2_RECEIVED_I2NP_MSGIDS_CLEANUP_TIMEOUT = 10 ; // in seconds
const int SSU2_DECAY_INTERVAL = 20 ; // in seconds
2022-07-29 18:45:02 -04:00
const size_t SSU2_MIN_WINDOW_SIZE = 16 ; // in packets
const size_t SSU2_MAX_WINDOW_SIZE = 256 ; // in packets
const size_t SSU2_MIN_RTO = 100 ; // in milliseconds
2024-03-15 09:06:33 +02:00
const size_t SSU2_INITIAL_RTO = 540 ; // in milliseconds
2022-07-29 18:45:02 -04:00
const size_t SSU2_MAX_RTO = 2500 ; // in milliseconds
2024-03-15 09:06:33 +02:00
const double SSU2_UNKNOWN_RTT = - 1 ;
const double SSU2_RTT_EWMA_ALPHA = 0.125 ;
2022-07-31 09:45:18 -04:00
const float SSU2_kAPPA = 1.8 ;
2023-02-03 19:17:46 -05:00
const int SSU2_MAX_NUM_ACNT = 255 ; // acnt, acks or nacks
2023-03-02 16:18:25 -05:00
const int SSU2_MAX_NUM_ACK_PACKETS = 511 ; // ackthrough + acnt + 1 range
2022-06-11 10:15:27 -04:00
const int SSU2_MAX_NUM_ACK_RANGES = 32 ; // to send
2022-08-03 16:06:07 -04:00
const uint8_t SSU2_MAX_NUM_FRAGMENTS = 64 ;
2023-03-02 16:18:25 -05:00
const int SSU2_SEND_DATETIME_NUM_PACKETS = 256 ;
2024-10-26 15:30:48 -04:00
const int SSU2_MIN_RELAY_RESPONSE_RESEND_VERSION = MAKE_VERSION_NUMBER ( 0 , 9 , 64 ) ; // 0.9.64
2023-02-11 09:41:51 +03:00
2022-09-01 18:40:54 -04:00
// flags
2022-10-09 20:24:43 +03:00
const uint8_t SSU2_FLAG_IMMEDIATE_ACK_REQUESTED = 0x01 ;
2022-06-05 19:33:36 -04:00
enum SSU2MessageType
{
eSSU2SessionRequest = 0 ,
eSSU2SessionCreated = 1 ,
eSSU2SessionConfirmed = 2 ,
eSSU2Data = 6 ,
eSSU2PeerTest = 7 ,
eSSU2Retry = 9 ,
eSSU2TokenRequest = 10 ,
eSSU2HolePunch = 11
} ;
enum SSU2BlockType
{
eSSU2BlkDateTime = 0 ,
eSSU2BlkOptions , // 1
eSSU2BlkRouterInfo , // 2
eSSU2BlkI2NPMessage , // 3
eSSU2BlkFirstFragment , // 4
eSSU2BlkFollowOnFragment , // 5
eSSU2BlkTermination , // 6
eSSU2BlkRelayRequest , // 7
eSSU2BlkRelayResponse , // 8
eSSU2BlkRelayIntro , // 9
eSSU2BlkPeerTest , // 10
eSSU2BlkNextNonce , // 11
eSSU2BlkAck , // 12
eSSU2BlkAddress , // 13
eSSU2BlkIntroKey , // 14
eSSU2BlkRelayTagRequest , // 15
eSSU2BlkRelayTag , // 16
eSSU2BlkNewToken , // 17
eSSU2BlkPathChallenge , // 18
eSSU2BlkPathResponse , // 19
eSSU2BlkFirstPacketNumber , // 20
eSSU2BlkPadding = 254
} ;
enum SSU2SessionState
{
eSSU2SessionStateUnknown ,
2022-06-22 13:58:22 -04:00
eSSU2SessionStateTokenReceived ,
2022-06-24 13:07:02 -04:00
eSSU2SessionStateSessionRequestSent ,
2022-07-13 12:45:20 -04:00
eSSU2SessionStateSessionRequestReceived ,
2022-06-24 13:07:02 -04:00
eSSU2SessionStateSessionCreatedSent ,
2022-08-08 19:57:48 -04:00
eSSU2SessionStateSessionCreatedReceived ,
2022-06-24 22:06:30 -04:00
eSSU2SessionStateSessionConfirmedSent ,
2022-06-05 19:33:36 -04:00
eSSU2SessionStateEstablished ,
2022-07-08 19:06:09 -04:00
eSSU2SessionStateClosing ,
2022-12-06 21:40:33 -05:00
eSSU2SessionStateClosingConfirmed ,
2022-06-05 19:33:36 -04:00
eSSU2SessionStateTerminated ,
2022-06-21 12:09:58 -04:00
eSSU2SessionStateFailed ,
eSSU2SessionStateIntroduced ,
2024-10-16 08:28:25 -04:00
eSSU2SessionStateHolePunch ,
2022-06-21 12:09:58 -04:00
eSSU2SessionStatePeerTest ,
2022-08-25 18:48:26 -04:00
eSSU2SessionStateTokenRequestReceived
2022-06-05 19:33:36 -04:00
} ;
enum SSU2PeerTestCode
{
eSSU2PeerTestCodeAccept = 0 ,
eSSU2PeerTestCodeBobReasonUnspecified = 1 ,
eSSU2PeerTestCodeBobNoCharlieAvailable = 2 ,
eSSU2PeerTestCodeBobLimitExceeded = 3 ,
eSSU2PeerTestCodeBobSignatureFailure = 4 ,
eSSU2PeerTestCodeCharlieReasonUnspecified = 64 ,
eSSU2PeerTestCodeCharlieUnsupportedAddress = 65 ,
eSSU2PeerTestCodeCharlieLimitExceeded = 66 ,
eSSU2PeerTestCodeCharlieSignatureFailure = 67 ,
eSSU2PeerTestCodeCharlieAliceIsAlreadyConnected = 68 ,
eSSU2PeerTestCodeCharlieAliceIsBanned = 69 ,
eSSU2PeerTestCodeCharlieAliceIsUnknown = 70 ,
eSSU2PeerTestCodeUnspecified = 128
2022-10-09 20:24:43 +03:00
} ;
2022-06-05 19:33:36 -04:00
2022-06-17 19:55:58 -04:00
enum SSU2RelayResponseCode
{
eSSU2RelayResponseCodeAccept = 0 ,
2022-06-21 15:45:35 -04:00
eSSU2RelayResponseCodeBobRelayTagNotFound = 5 ,
2022-06-21 16:20:39 -04:00
eSSU2RelayResponseCodeCharlieUnsupportedAddress = 65 ,
2022-06-21 15:45:35 -04:00
eSSU2RelayResponseCodeCharlieSignatureFailure = 67 ,
eSSU2RelayResponseCodeCharlieAliceIsUnknown = 70
2022-10-09 20:24:43 +03:00
} ;
2022-07-09 17:05:23 -04:00
enum SSU2TerminationReason
{
eSSU2TerminationReasonNormalClose = 0 ,
eSSU2TerminationReasonTerminationReceived = 1 ,
eSSU2TerminationReasonIdleTimeout = 2 ,
eSSU2TerminationReasonRouterShutdown = 3 ,
eSSU2TerminationReasonDataPhaseAEADFailure = 4 ,
eSSU2TerminationReasonIncompatibleOptions = 5 ,
eSSU2TerminationReasonTncompatibleSignatureType = 6 ,
eSSU2TerminationReasonClockSkew = 7 ,
eSSU2TerminationPaddingViolation = 8 ,
eSSU2TerminationReasonAEADFramingError = 9 ,
eSSU2TerminationReasonPayloadFormatError = 10 ,
eSSU2TerminationReasonSessionRequestError = 11 ,
eSSU2TerminationReasonSessionCreatedError = 12 ,
eSSU2TerminationReasonSessionConfirmedError = 13 ,
eSSU2TerminationReasonTimeout = 14 ,
eSSU2TerminationReasonRouterInfoSignatureVerificationFail = 15 ,
eSSU2TerminationReasonInvalidS = 16 ,
eSSU2TerminationReasonBanned = 17 ,
eSSU2TerminationReasonBadToken = 18 ,
eSSU2TerminationReasonConnectionLimits = 19 ,
eSSU2TerminationReasonIncompatibleVersion = 20 ,
2022-08-02 13:35:18 -04:00
eSSU2TerminationReasonWrongNetID = 21 ,
eSSU2TerminationReasonReplacedByNewSession = 22
2022-10-09 20:24:43 +03:00
} ;
2022-06-05 19:33:36 -04:00
struct SSU2IncompleteMessage
{
struct Fragment
{
2022-07-14 20:12:27 -04:00
uint8_t buf [ SSU2_MAX_PACKET_SIZE ] ;
2022-06-05 19:33:36 -04:00
size_t len ;
2023-01-16 21:40:23 -05:00
int fragmentNum ;
2022-06-05 19:33:36 -04:00
bool isLast ;
2023-01-16 21:40:23 -05:00
std : : shared_ptr < Fragment > next ;
2022-06-05 19:33:36 -04:00
} ;
std : : shared_ptr < I2NPMessage > msg ;
int nextFragmentNum ;
uint32_t lastFragmentInsertTime ; // in seconds
2023-01-16 21:40:23 -05:00
std : : shared_ptr < Fragment > outOfSequenceFragments ; // #1 and more
2022-08-03 16:06:07 -04:00
void AttachNextFragment ( const uint8_t * fragment , size_t fragmentSize ) ;
2023-01-15 22:50:54 -05:00
bool ConcatOutOfSequenceFragments ( ) ; // true if message complete
2023-01-16 21:40:23 -05:00
void AddOutOfSequenceFragment ( std : : shared_ptr < Fragment > fragment ) ;
2022-06-05 19:33:36 -04:00
} ;
2022-08-04 18:13:44 -04:00
struct SSU2SentPacket
{
uint8_t payload [ SSU2_MAX_PACKET_SIZE ] ;
size_t payloadSize = 0 ;
uint64_t sendTime ; // in milliseconds
int numResends = 0 ;
} ;
2022-10-09 20:24:43 +03:00
2022-06-05 19:33:36 -04:00
// RouterInfo flags
const uint8_t SSU2_ROUTER_INFO_FLAG_REQUEST_FLOOD = 0x01 ;
const uint8_t SSU2_ROUTER_INFO_FLAG_GZIP = 0x02 ;
class SSU2Server ;
class SSU2Session : public TransportSession , public std : : enable_shared_from_this < SSU2Session >
{
2024-09-22 17:22:08 -04:00
protected :
union Header
2022-06-05 19:33:36 -04:00
{
2024-09-22 17:22:08 -04:00
uint64_t ll [ 2 ] ;
uint8_t buf [ 16 ] ;
struct
{
uint64_t connID ;
uint32_t packetNum ;
uint8_t type ;
uint8_t flags [ 3 ] ;
} h ;
} ;
2022-06-05 19:33:36 -04:00
2024-09-22 17:22:08 -04:00
private :
struct HandshakePacket
{
Header header ;
uint8_t headerX [ 48 ] ; // part1 for SessionConfirmed
uint8_t payload [ SSU2_MAX_PACKET_SIZE * 2 ] ;
size_t payloadSize = 0 ;
uint64_t sendTime = 0 ; // in milliseconds
bool isSecondFragment = false ; // for SessionConfirmed
} ;
2022-06-05 19:33:36 -04:00
2024-09-22 17:22:08 -04:00
typedef std : : function < void ( ) > OnEstablished ;
2022-06-05 19:33:36 -04:00
public :
SSU2Session ( SSU2Server & server , std : : shared_ptr < const i2p : : data : : RouterInfo > in_RemoteRouter = nullptr ,
2024-09-22 18:02:12 -04:00
std : : shared_ptr < const i2p : : data : : RouterInfo : : Address > addr = nullptr , bool noise = true ) ;
2024-09-22 17:22:08 -04:00
virtual ~ SSU2Session ( ) ;
2022-06-05 19:33:36 -04:00
void SetRemoteEndpoint ( const boost : : asio : : ip : : udp : : endpoint & ep ) { m_RemoteEndpoint = ep ; } ;
const boost : : asio : : ip : : udp : : endpoint & GetRemoteEndpoint ( ) const { return m_RemoteEndpoint ; } ;
2022-06-09 18:04:37 -04:00
i2p : : data : : RouterInfo : : CompatibleTransports GetRemoteTransports ( ) const { return m_RemoteTransports ; } ;
2024-03-01 21:59:52 -05:00
i2p : : data : : RouterInfo : : CompatibleTransports GetRemotePeerTestTransports ( ) const { return m_RemotePeerTestTransports ; } ;
2022-06-11 21:26:23 -04:00
std : : shared_ptr < const i2p : : data : : RouterInfo : : Address > GetAddress ( ) const { return m_Address ; } ;
2022-06-05 19:33:36 -04:00
void SetOnEstablished ( OnEstablished e ) { m_OnEstablished = e ; } ;
2022-06-22 11:59:29 -04:00
OnEstablished GetOnEstablished ( ) const { return m_OnEstablished ; } ;
2022-06-05 19:33:36 -04:00
2024-09-24 20:03:15 -04:00
virtual void Connect ( ) ;
2022-06-05 19:33:36 -04:00
bool Introduce ( std : : shared_ptr < SSU2Session > session , uint32_t relayTag ) ;
2022-06-19 16:40:03 -04:00
void WaitForIntroduction ( ) ;
2022-06-06 17:28:39 -04:00
void SendPeerTest ( ) ; // Alice, Data message
2022-07-10 17:13:25 -04:00
void SendKeepAlive ( ) ;
2022-07-09 17:05:23 -04:00
void RequestTermination ( SSU2TerminationReason reason ) ;
2022-06-05 19:33:36 -04:00
void CleanUp ( uint64_t ts ) ;
void FlushData ( ) ;
void Done ( ) override ;
2022-06-13 14:02:36 -04:00
void SendLocalRouterInfo ( bool update ) override ;
2024-10-28 20:36:50 -04:00
void SendI2NPMessages ( std : : list < std : : shared_ptr < I2NPMessage > > & msgs ) override ;
2024-06-01 17:46:18 -04:00
void MoveSendQueue ( std : : shared_ptr < SSU2Session > other ) ;
2022-06-17 18:45:37 -04:00
uint32_t GetRelayTag ( ) const override { return m_RelayTag ; } ;
2024-05-14 18:17:47 -04:00
size_t Resend ( uint64_t ts ) ; // return number of resent packets
uint64_t GetLastResendTime ( ) const { return m_LastResendTime ; } ;
2022-12-18 22:37:26 +03:00
bool IsEstablished ( ) const override { return m_State = = eSSU2SessionStateEstablished ; } ;
2024-12-16 19:49:14 -05:00
i2p : : data : : RouterInfo : : SupportedTransports GetTransportType ( ) const override ;
2022-06-05 19:33:36 -04:00
uint64_t GetConnID ( ) const { return m_SourceConnID ; } ;
SSU2SessionState GetState ( ) const { return m_State ; } ;
void SetState ( SSU2SessionState state ) { m_State = state ; } ;
2024-09-24 20:03:15 -04:00
virtual bool ProcessFirstIncomingMessage ( uint64_t connID , uint8_t * buf , size_t len ) ;
2022-06-05 19:33:36 -04:00
bool ProcessSessionCreated ( uint8_t * buf , size_t len ) ;
bool ProcessSessionConfirmed ( uint8_t * buf , size_t len ) ;
bool ProcessRetry ( uint8_t * buf , size_t len ) ;
bool ProcessHolePunch ( uint8_t * buf , size_t len ) ;
2024-09-22 17:22:08 -04:00
virtual bool ProcessPeerTest ( uint8_t * buf , size_t len ) ;
2022-09-07 19:11:33 -04:00
void ProcessData ( uint8_t * buf , size_t len , const boost : : asio : : ip : : udp : : endpoint & from ) ;
2022-06-05 19:33:36 -04:00
2024-09-22 17:22:08 -04:00
protected :
2024-09-22 20:25:41 -04:00
SSU2Server & GetServer ( ) { return m_Server ; }
RouterStatus GetRouterStatus ( ) const ;
void SetRouterStatus ( RouterStatus status ) const ;
2024-09-24 14:37:27 -04:00
size_t GetMaxPayloadSize ( ) const { return m_MaxPayloadSize ; }
2024-09-30 18:27:13 -04:00
void SetIsDataReceived ( bool dataReceived ) { m_IsDataReceived = dataReceived ; } ;
2024-09-22 20:25:41 -04:00
uint64_t GetSourceConnID ( ) const { return m_SourceConnID ; }
2024-09-22 17:22:08 -04:00
void SetSourceConnID ( uint64_t sourceConnID ) { m_SourceConnID = sourceConnID ; }
2024-09-22 20:25:41 -04:00
uint64_t GetDestConnID ( ) const { return m_DestConnID ; }
2024-09-22 17:22:08 -04:00
void SetDestConnID ( uint64_t destConnID ) { m_DestConnID = destConnID ; }
2024-09-24 14:37:27 -04:00
void SetAddress ( std : : shared_ptr < const i2p : : data : : RouterInfo : : Address > addr ) { m_Address = addr ; }
2024-09-22 17:22:08 -04:00
void HandlePayload ( const uint8_t * buf , size_t len ) ;
2024-09-24 14:37:27 -04:00
size_t CreateAddressBlock ( uint8_t * buf , size_t len , const boost : : asio : : ip : : udp : : endpoint & ep ) ;
size_t CreatePaddingBlock ( uint8_t * buf , size_t len , size_t minSize = 0 ) ;
size_t CreatePeerTestBlock ( uint8_t * buf , size_t len , uint8_t msg , SSU2PeerTestCode code , const uint8_t * routerHash , const uint8_t * signedData , size_t signedDataLen ) ;
2024-10-17 21:09:37 -04:00
bool ExtractEndpoint ( const uint8_t * buf , size_t size , boost : : asio : : ip : : udp : : endpoint & ep ) ;
2024-09-22 17:22:08 -04:00
2022-06-05 19:33:36 -04:00
private :
2022-07-26 19:56:30 -04:00
void Terminate ( ) ;
2022-06-05 19:33:36 -04:00
void Established ( ) ;
2022-06-19 16:40:03 -04:00
void ScheduleConnectTimer ( ) ;
void HandleConnectTimer ( const boost : : system : : error_code & ecode ) ;
2024-10-28 20:36:50 -04:00
void PostI2NPMessages ( ) ;
2022-07-26 13:00:41 -04:00
bool SendQueue ( ) ; // returns true if ack block was sent
bool SendFragmentedMessage ( std : : shared_ptr < I2NPMessage > msg ) ;
2022-07-19 16:09:16 -04:00
void ResendHandshakePacket ( ) ;
2022-08-08 13:08:12 -04:00
void ConnectAfterIntroduction ( ) ;
2022-10-09 20:24:43 +03:00
2022-06-05 19:33:36 -04:00
void ProcessSessionRequest ( Header & header , uint8_t * buf , size_t len ) ;
void ProcessTokenRequest ( Header & header , uint8_t * buf , size_t len ) ;
void SendSessionRequest ( uint64_t token = 0 ) ;
void SendSessionCreated ( const uint8_t * X ) ;
void SendSessionConfirmed ( const uint8_t * Y ) ;
void KDFDataPhase ( uint8_t * keydata_ab , uint8_t * keydata_ba ) ;
void SendTokenRequest ( ) ;
void SendRetry ( ) ;
2022-09-01 18:40:54 -04:00
uint32_t SendData ( const uint8_t * buf , size_t len , uint8_t flags = 0 ) ; // returns packet num
2022-06-05 19:33:36 -04:00
void SendQuickAck ( ) ;
void SendTermination ( ) ;
2022-08-02 13:35:18 -04:00
void SendPathResponse ( const uint8_t * data , size_t len ) ;
2022-09-07 19:11:33 -04:00
void SendPathChallenge ( ) ;
2022-10-09 20:24:43 +03:00
2022-08-08 19:57:48 -04:00
void HandleDateTime ( const uint8_t * buf , size_t len ) ;
2024-06-13 18:10:45 -04:00
void HandleRouterInfo ( const uint8_t * buf , size_t len ) ;
2022-06-05 19:33:36 -04:00
void HandleAck ( const uint8_t * buf , size_t len ) ;
2022-07-29 15:24:24 -04:00
void HandleAckRange ( uint32_t firstPacketNum , uint32_t lastPacketNum , uint64_t ts ) ;
2024-10-17 21:09:37 -04:00
virtual void HandleAddress ( const uint8_t * buf , size_t len ) ;
2022-06-05 19:33:36 -04:00
size_t CreateEndpoint ( uint8_t * buf , size_t len , const boost : : asio : : ip : : udp : : endpoint & ep ) ;
std : : shared_ptr < const i2p : : data : : RouterInfo : : Address > FindLocalAddress ( ) const ;
2022-07-13 19:35:18 -04:00
void AdjustMaxPayloadSize ( ) ;
2023-07-30 15:29:10 +03:00
bool GetTestingState ( ) const ;
void SetTestingState ( bool testing ) const ;
2022-06-05 19:33:36 -04:00
std : : shared_ptr < const i2p : : data : : RouterInfo > ExtractRouterInfo ( const uint8_t * buf , size_t size ) ;
bool UpdateReceivePacketNum ( uint32_t packetNum ) ; // for Ack, returns false if duplicate
void HandleFirstFragment ( const uint8_t * buf , size_t len ) ;
void HandleFollowOnFragment ( const uint8_t * buf , size_t len ) ;
void HandleRelayRequest ( const uint8_t * buf , size_t len ) ;
2022-12-19 13:28:21 -05:00
void HandleRelayIntro ( const uint8_t * buf , size_t len , int attempts = 0 ) ;
2022-06-05 19:33:36 -04:00
void HandleRelayResponse ( const uint8_t * buf , size_t len ) ;
2024-09-22 20:25:41 -04:00
virtual void HandlePeerTest ( const uint8_t * buf , size_t len ) ;
2023-01-05 15:33:41 -05:00
void HandleI2NPMsg ( std : : shared_ptr < I2NPMessage > & & msg ) ;
2023-02-11 09:41:51 +03:00
2022-06-05 19:33:36 -04:00
size_t CreateRouterInfoBlock ( uint8_t * buf , size_t len , std : : shared_ptr < const i2p : : data : : RouterInfo > r ) ;
2024-05-26 15:33:37 -04:00
size_t CreateRouterInfoBlock ( uint8_t * buf , size_t len , std : : shared_ptr < const i2p : : data : : RouterInfo : : Buffer > riBuffer ) ;
2022-06-05 19:33:36 -04:00
size_t CreateAckBlock ( uint8_t * buf , size_t len ) ;
size_t CreateI2NPBlock ( uint8_t * buf , size_t len , std : : shared_ptr < I2NPMessage > & & msg ) ;
size_t CreateFirstFragmentBlock ( uint8_t * buf , size_t len , std : : shared_ptr < I2NPMessage > msg ) ;
size_t CreateFollowOnFragmentBlock ( uint8_t * buf , size_t len , std : : shared_ptr < I2NPMessage > msg , uint8_t & fragmentNum , uint32_t msgID ) ;
size_t CreateRelayIntroBlock ( uint8_t * buf , size_t len , const uint8_t * introData , size_t introDataLen ) ;
2024-10-16 12:07:13 -04:00
size_t CreateRelayResponseBlock ( uint8_t * buf , size_t len , SSU2RelayResponseCode code , uint32_t nonce , uint64_t token , bool v4 ) ;
2022-06-05 19:33:36 -04:00
size_t CreatePeerTestBlock ( uint8_t * buf , size_t len , uint32_t nonce ) ; // Alice
2022-07-09 17:05:23 -04:00
size_t CreateTerminationBlock ( uint8_t * buf , size_t len ) ;
2024-09-22 17:22:08 -04:00
2022-06-05 19:33:36 -04:00
private :
SSU2Server & m_Server ;
std : : shared_ptr < i2p : : crypto : : X25519Keys > m_EphemeralKeys ;
std : : unique_ptr < i2p : : crypto : : NoiseSymmetricState > m_NoiseState ;
2022-07-12 19:04:03 -04:00
std : : unique_ptr < HandshakePacket > m_SessionConfirmedFragment ; // for Bob if applicable or second fragment for Alice
std : : unique_ptr < HandshakePacket > m_SentHandshakePacket ; // SessionRequest, SessionCreated or SessionConfirmed
2022-06-05 19:33:36 -04:00
std : : shared_ptr < const i2p : : data : : RouterInfo : : Address > m_Address ;
boost : : asio : : ip : : udp : : endpoint m_RemoteEndpoint ;
2024-03-01 21:59:52 -05:00
i2p : : data : : RouterInfo : : CompatibleTransports m_RemoteTransports , m_RemotePeerTestTransports ;
2024-10-26 15:30:48 -04:00
int m_RemoteVersion ;
2022-06-05 19:33:36 -04:00
uint64_t m_DestConnID , m_SourceConnID ;
SSU2SessionState m_State ;
uint8_t m_KeyDataSend [ 64 ] , m_KeyDataReceive [ 64 ] ;
2023-02-11 20:11:34 -05:00
uint32_t m_SendPacketNum , m_ReceivePacketNum , m_LastDatetimeSentPacketNum ;
2022-06-05 19:33:36 -04:00
std : : set < uint32_t > m_OutOfSequencePackets ; // packet nums > receive packet num
2022-08-04 18:13:44 -04:00
std : : map < uint32_t , std : : shared_ptr < SSU2SentPacket > > m_SentPackets ; // packetNum -> packet
2023-01-17 21:32:36 -05:00
std : : unordered_map < uint32_t , std : : shared_ptr < SSU2IncompleteMessage > > m_IncompleteMessages ; // msgID -> I2NP
2024-09-22 17:22:08 -04:00
std : : unordered_map < uint32_t , std : : pair < std : : shared_ptr < SSU2Session > , uint64_t > > m_RelaySessions ; // nonce->(Alice, timestamp) for Bob or nonce->(Charlie, timestamp) for Alice
2022-06-05 19:33:36 -04:00
std : : list < std : : shared_ptr < I2NPMessage > > m_SendQueue ;
i2p : : I2NPMessagesHandler m_Handler ;
2024-10-28 20:36:50 -04:00
std : : list < std : : shared_ptr < I2NPMessage > > m_IntermediateQueue ; // from transports
mutable std : : mutex m_IntermediateQueueMutex ;
2022-06-05 19:33:36 -04:00
bool m_IsDataReceived ;
2024-03-15 09:06:33 +02:00
double m_RTT ;
2024-03-26 09:41:49 +02:00
int m_MsgLocalExpirationTimeout ;
int m_MsgLocalSemiExpirationTimeout ;
2024-03-15 09:06:33 +02:00
size_t m_WindowSize , m_RTO ;
2022-06-05 19:33:36 -04:00
uint32_t m_RelayTag ; // between Bob and Charlie
OnEstablished m_OnEstablished ; // callback from Established
2022-06-19 16:40:03 -04:00
boost : : asio : : deadline_timer m_ConnectTimer ;
2022-07-09 17:05:23 -04:00
SSU2TerminationReason m_TerminationReason ;
2022-07-13 19:35:18 -04:00
size_t m_MaxPayloadSize ;
2022-09-14 19:08:14 -04:00
std : : unique_ptr < i2p : : data : : IdentHash > m_PathChallenge ;
2023-01-05 15:33:41 -05:00
std : : unordered_map < uint32_t , uint32_t > m_ReceivedI2NPMsgIDs ; // msgID -> timestamp in seconds
2024-08-14 10:13:35 -04:00
uint64_t m_LastResendTime , m_LastResendAttemptTime ; // in milliseconds
2025-01-03 22:04:09 -05:00
int m_NumRanges ;
uint8_t m_Ranges [ SSU2_MAX_NUM_ACK_RANGES * 2 ] ; // ranges sent with previous Ack if any
2022-06-05 19:33:36 -04:00
} ;
2024-09-22 17:22:08 -04:00
2022-06-05 19:33:36 -04:00
inline uint64_t CreateHeaderMask ( const uint8_t * kh , const uint8_t * nonce )
{
uint64_t data = 0 ;
i2p : : crypto : : ChaCha20 ( ( uint8_t * ) & data , 8 , kh , nonce , ( uint8_t * ) & data ) ;
return data ;
}
2024-10-13 19:53:40 -04:00
inline void CreateNonce ( uint64_t seqn , uint8_t * nonce )
{
memset ( nonce , 0 , 4 ) ;
htole64buf ( nonce + 4 , seqn ) ;
}
2022-06-05 19:33:36 -04:00
}
}
# endif