2014-12-31 15:14:53 +01:00
# include <string.h>
2014-01-15 02:57:33 +01:00
# include "I2PEndian.h"
2015-11-03 15:15:49 +01:00
# include "Crypto.h"
2013-11-25 00:10:27 +01:00
# include "Log.h"
2014-01-15 02:57:33 +01:00
# include "Timestamp.h"
2017-04-22 02:04:16 +02:00
# include "NetDb.hpp"
2016-05-25 20:17:34 +02:00
# include "Tunnel.h"
2013-11-25 00:10:27 +01:00
# include "LeaseSet.h"
namespace i2p
{
namespace data
{
2018-01-06 04:48:51 +01:00
2019-01-09 18:47:47 +01:00
LeaseSet : : LeaseSet ( bool storeLeases ) :
2019-01-14 19:49:27 +01:00
m_IsValid ( false ) , m_StoreLeases ( storeLeases ) , m_ExpirationTime ( 0 ) , m_EncryptionKey ( nullptr ) ,
m_Buffer ( nullptr ) , m_BufferLen ( 0 )
2018-12-21 21:00:03 +01:00
{
}
2016-02-08 01:45:06 +01:00
LeaseSet : : LeaseSet ( const uint8_t * buf , size_t len , bool storeLeases ) :
2019-01-14 19:49:27 +01:00
m_IsValid ( true ) , m_StoreLeases ( storeLeases ) , m_ExpirationTime ( 0 ) , m_EncryptionKey ( nullptr )
2013-11-25 00:10:27 +01:00
{
2015-04-08 15:39:02 +02:00
m_Buffer = new uint8_t [ len ] ;
2014-07-29 19:44:54 +02:00
memcpy ( m_Buffer , buf , len ) ;
m_BufferLen = len ;
ReadFromBuffer ( ) ;
}
2018-01-25 16:32:08 +01:00
void LeaseSet : : Update ( const uint8_t * buf , size_t len , bool verifySignature )
2018-01-06 04:48:51 +01:00
{
2015-04-08 15:39:02 +02:00
if ( len > m_BufferLen )
{
auto oldBuffer = m_Buffer ;
m_Buffer = new uint8_t [ len ] ;
delete [ ] oldBuffer ;
2018-01-06 04:48:51 +01:00
}
2014-07-29 19:44:54 +02:00
memcpy ( m_Buffer , buf , len ) ;
m_BufferLen = len ;
2018-01-25 16:32:08 +01:00
ReadFromBuffer ( false , verifySignature ) ;
2014-07-22 02:14:11 +02:00
}
2016-02-08 01:45:06 +01:00
void LeaseSet : : PopulateLeases ( )
{
m_StoreLeases = true ;
ReadFromBuffer ( false ) ;
2018-01-06 04:48:51 +01:00
}
2018-01-25 16:09:34 +01:00
void LeaseSet : : ReadFromBuffer ( bool readIdentity , bool verifySignature )
2018-01-06 04:48:51 +01:00
{
2015-11-03 15:15:49 +01:00
if ( readIdentity | | ! m_Identity )
m_Identity = std : : make_shared < IdentityEx > ( m_Buffer , m_BufferLen ) ;
size_t size = m_Identity - > GetFullLen ( ) ;
2016-02-02 17:55:38 +01:00
if ( size > m_BufferLen )
{
LogPrint ( eLogError , " LeaseSet: identity length " , size , " exceeds buffer size " , m_BufferLen ) ;
m_IsValid = false ;
return ;
}
2019-01-14 19:49:27 +01:00
if ( m_StoreLeases )
{
if ( ! m_EncryptionKey ) m_EncryptionKey = new uint8_t [ 256 ] ;
memcpy ( m_EncryptionKey , m_Buffer + size , 256 ) ;
}
2014-08-22 03:57:24 +02:00
size + = 256 ; // encryption key
2015-11-03 15:15:49 +01:00
size + = m_Identity - > GetSigningPublicKeyLen ( ) ; // unused signing key
2014-08-22 03:57:24 +02:00
uint8_t num = m_Buffer [ size ] ;
size + + ; // num
2015-12-18 15:08:23 +01:00
LogPrint ( eLogDebug , " LeaseSet: read num= " , ( int ) num ) ;
2016-02-02 17:55:38 +01:00
if ( ! num | | num > MAX_NUM_LEASES )
2018-01-06 04:48:51 +01:00
{
2016-02-02 17:55:38 +01:00
LogPrint ( eLogError , " LeaseSet: incorrect number of leases " , ( int ) num ) ;
m_IsValid = false ;
return ;
2018-01-06 04:48:51 +01:00
}
2013-11-25 00:10:27 +01:00
2019-01-09 18:47:47 +01:00
UpdateLeasesBegin ( ) ;
2016-02-09 21:27:23 +01:00
2014-01-26 15:07:31 +01:00
// process leases
2016-02-08 01:45:06 +01:00
m_ExpirationTime = 0 ;
2016-02-10 04:42:01 +01:00
auto ts = i2p : : util : : GetMillisecondsSinceEpoch ( ) ;
2014-08-22 03:57:24 +02:00
const uint8_t * leases = m_Buffer + size ;
for ( int i = 0 ; i < num ; i + + )
2013-11-25 00:10:27 +01:00
{
2014-12-31 15:14:53 +01:00
Lease lease ;
2015-03-23 17:55:42 +01:00
lease . tunnelGateway = leases ;
leases + = 32 ; // gateway
lease . tunnelID = bufbe32toh ( leases ) ;
leases + = 4 ; // tunnel ID
2018-01-06 04:48:51 +01:00
lease . endDate = bufbe64toh ( leases ) ;
2015-03-23 17:55:42 +01:00
leases + = 8 ; // end date
2019-01-02 21:40:48 +01:00
UpdateLease ( lease , ts ) ;
2018-01-06 04:48:51 +01:00
}
2016-02-15 00:30:07 +01:00
if ( ! m_ExpirationTime )
2016-02-10 04:42:01 +01:00
{
LogPrint ( eLogWarning , " LeaseSet: all leases are expired. Dropped " ) ;
m_IsValid = false ;
return ;
2018-01-06 04:48:51 +01:00
}
2016-02-15 00:30:07 +01:00
m_ExpirationTime + = LEASE_ENDDATE_THRESHOLD ;
2019-01-09 18:47:47 +01:00
UpdateLeasesEnd ( ) ;
// verify
if ( verifySignature & & ! m_Identity - > Verify ( m_Buffer , leases - m_Buffer , leases ) )
{
LogPrint ( eLogWarning , " LeaseSet: verification failed " ) ;
m_IsValid = false ;
}
}
void LeaseSet : : UpdateLeasesBegin ( )
{
// reset existing leases
if ( m_StoreLeases )
for ( auto & it : m_Leases )
it - > isUpdated = false ;
else
m_Leases . clear ( ) ;
}
void LeaseSet : : UpdateLeasesEnd ( )
{
2018-01-06 04:48:51 +01:00
// delete old leases
2016-02-09 21:27:23 +01:00
if ( m_StoreLeases )
2018-01-06 04:48:51 +01:00
{
2016-02-09 21:27:23 +01:00
for ( auto it = m_Leases . begin ( ) ; it ! = m_Leases . end ( ) ; )
2018-01-06 04:48:51 +01:00
{
2016-02-09 21:27:23 +01:00
if ( ! ( * it ) - > isUpdated )
{
( * it ) - > endDate = 0 ; // somebody might still hold it
m_Leases . erase ( it + + ) ;
2018-01-06 04:48:51 +01:00
}
2016-02-09 21:27:23 +01:00
else
2016-08-05 20:23:54 +02:00
+ + it ;
2016-02-09 21:27:23 +01:00
}
}
2018-01-06 04:48:51 +01:00
}
2016-02-17 04:57:38 +01:00
2019-01-02 21:40:48 +01:00
void LeaseSet : : UpdateLease ( const Lease & lease , uint64_t ts )
{
if ( ts < lease . endDate + LEASE_ENDDATE_THRESHOLD )
{
if ( lease . endDate > m_ExpirationTime )
m_ExpirationTime = lease . endDate ;
if ( m_StoreLeases )
{
auto ret = m_Leases . insert ( std : : make_shared < Lease > ( lease ) ) ;
if ( ! ret . second ) ( * ret . first ) - > endDate = lease . endDate ; // update existing
( * ret . first ) - > isUpdated = true ;
// check if lease's gateway is in our netDb
if ( ! netdb . FindRouter ( lease . tunnelGateway ) )
{
// if not found request it
LogPrint ( eLogInfo , " LeaseSet: Lease's tunnel gateway not found, requesting " ) ;
netdb . RequestDestination ( lease . tunnelGateway ) ;
}
}
}
else
LogPrint ( eLogWarning , " LeaseSet: Lease is expired already " ) ;
}
2018-01-26 20:33:06 +01:00
uint64_t LeaseSet : : ExtractTimestamp ( const uint8_t * buf , size_t len ) const
2016-02-17 04:57:38 +01:00
{
if ( ! m_Identity ) return 0 ;
size_t size = m_Identity - > GetFullLen ( ) ;
if ( size > len ) return 0 ;
size + = 256 ; // encryption key
size + = m_Identity - > GetSigningPublicKeyLen ( ) ; // unused signing key
if ( size > len ) return 0 ;
2018-01-26 20:33:06 +01:00
uint8_t num = buf [ size ] ;
2016-02-17 04:57:38 +01:00
size + + ; // num
2016-05-25 20:17:34 +02:00
if ( size + num * LEASE_SIZE > len ) return 0 ;
2016-02-17 04:57:38 +01:00
uint64_t timestamp = 0 ;
for ( int i = 0 ; i < num ; i + + )
{
size + = 36 ; // gateway (32) + tunnelId(4)
2018-01-06 04:48:51 +01:00
auto endDate = bufbe64toh ( buf + size ) ;
2016-02-17 04:57:38 +01:00
size + = 8 ; // end date
if ( ! timestamp | | endDate < timestamp )
timestamp = endDate ;
2018-01-06 04:48:51 +01:00
}
2016-02-17 04:57:38 +01:00
return timestamp ;
2018-01-06 04:48:51 +01:00
}
2016-02-17 04:57:38 +01:00
bool LeaseSet : : IsNewer ( const uint8_t * buf , size_t len ) const
{
2018-01-26 20:33:06 +01:00
return ExtractTimestamp ( buf , len ) > ExtractTimestamp ( m_Buffer , m_BufferLen ) ;
2018-01-06 04:48:51 +01:00
}
2016-07-22 15:56:17 +02:00
2016-09-03 17:46:47 +02:00
bool LeaseSet : : ExpiresSoon ( const uint64_t dlt , const uint64_t fudge ) const
2016-07-22 15:56:17 +02:00
{
auto now = i2p : : util : : GetMillisecondsSinceEpoch ( ) ;
2016-09-03 17:46:47 +02:00
if ( fudge ) now + = rand ( ) % fudge ;
2016-07-22 15:56:17 +02:00
if ( now > = m_ExpirationTime ) return true ;
return m_ExpirationTime - now < = dlt ;
}
2016-08-27 19:17:34 +02:00
const std : : vector < std : : shared_ptr < const Lease > > LeaseSet : : GetNonExpiredLeases ( bool withThreshold ) const
{
return GetNonExpiredLeasesExcluding ( [ ] ( const Lease & l ) - > bool { return false ; } , withThreshold ) ;
}
2018-01-06 04:48:51 +01:00
2016-08-27 19:17:34 +02:00
const std : : vector < std : : shared_ptr < const Lease > > LeaseSet : : GetNonExpiredLeasesExcluding ( LeaseInspectFunc exclude , bool withThreshold ) const
2014-01-15 02:57:33 +01:00
{
auto ts = i2p : : util : : GetMillisecondsSinceEpoch ( ) ;
2016-02-11 04:51:08 +01:00
std : : vector < std : : shared_ptr < const Lease > > leases ;
2016-08-05 20:23:54 +02:00
for ( const auto & it : m_Leases )
2015-03-26 15:30:29 +01:00
{
2016-02-09 16:46:27 +01:00
auto endDate = it - > endDate ;
2016-02-15 00:30:07 +01:00
if ( withThreshold )
endDate + = LEASE_ENDDATE_THRESHOLD ;
else
endDate - = LEASE_ENDDATE_THRESHOLD ;
2016-08-27 19:17:34 +02:00
if ( ts < endDate & & ! exclude ( * it ) )
2014-03-23 14:25:16 +01:00
leases . push_back ( it ) ;
2018-01-06 04:48:51 +01:00
}
return leases ;
}
2014-01-15 02:57:33 +01:00
bool LeaseSet : : HasExpiredLeases ( ) const
2018-01-06 05:01:44 +01:00
{
2014-01-15 02:57:33 +01:00
auto ts = i2p : : util : : GetMillisecondsSinceEpoch ( ) ;
2016-08-05 20:23:54 +02:00
for ( const auto & it : m_Leases )
2016-02-09 16:46:27 +01:00
if ( ts > = it - > endDate ) return true ;
2014-01-15 02:57:33 +01:00
return false ;
2018-01-06 05:01:44 +01:00
}
2014-01-15 02:57:33 +01:00
2016-02-08 01:45:06 +01:00
bool LeaseSet : : IsExpired ( ) const
2014-01-15 02:57:33 +01:00
{
2016-06-30 23:21:18 +02:00
if ( m_StoreLeases & & IsEmpty ( ) ) return true ;
2014-01-15 02:57:33 +01:00
auto ts = i2p : : util : : GetMillisecondsSinceEpoch ( ) ;
2016-02-08 01:45:06 +01:00
return ts > m_ExpirationTime ;
2016-05-25 20:17:34 +02:00
}
2017-11-08 02:30:05 +01:00
void LeaseSet : : Encrypt ( const uint8_t * data , uint8_t * encrypted , BN_CTX * ctx ) const
2017-11-07 21:05:22 +01:00
{
2019-01-14 19:49:27 +01:00
if ( ! m_EncryptionKey ) return ;
2017-11-07 21:05:22 +01:00
auto encryptor = m_Identity - > CreateEncryptor ( m_EncryptionKey ) ;
if ( encryptor )
2018-03-09 20:56:06 +01:00
encryptor - > Encrypt ( data , encrypted , ctx , true ) ;
2017-11-07 21:05:22 +01:00
}
2018-12-21 21:00:03 +01:00
void LeaseSet : : SetBuffer ( const uint8_t * buf , size_t len )
{
if ( m_Buffer ) delete [ ] m_Buffer ;
m_Buffer = new uint8_t [ len ] ;
m_BufferLen = len ;
memcpy ( m_Buffer , buf , len ) ;
}
2019-01-09 18:47:47 +01:00
LeaseSet2 : : LeaseSet2 ( uint8_t storeType , const uint8_t * buf , size_t len , bool storeLeases ) :
LeaseSet ( storeLeases ) , m_StoreType ( storeType )
2018-12-26 21:27:32 +01:00
{
2018-12-21 21:00:03 +01:00
SetBuffer ( buf , len ) ;
2019-01-08 17:26:50 +01:00
if ( storeType = = NETDB_STORE_TYPE_ENCRYPTED_LEASESET2 )
ReadFromBufferEncrypted ( buf , len ) ;
else
ReadFromBuffer ( buf , len ) ;
2018-12-21 21:00:03 +01:00
}
2019-01-17 01:00:17 +01:00
void LeaseSet2 : : Update ( const uint8_t * buf , size_t len , bool verifySignature )
{
// shouldn't be called for now. Must be called from NetDb::AddLeaseSet later
SetBuffer ( buf , len ) ;
// TODO:verify signature if requested
}
2018-12-26 21:27:32 +01:00
void LeaseSet2 : : ReadFromBuffer ( const uint8_t * buf , size_t len )
2018-12-21 21:00:03 +01:00
{
2019-01-02 20:19:10 +01:00
// standard LS2 header
2018-12-21 21:00:03 +01:00
auto identity = std : : make_shared < IdentityEx > ( buf , len ) ;
SetIdentity ( identity ) ;
size_t offset = identity - > GetFullLen ( ) ;
2019-01-08 17:26:50 +01:00
if ( offset + 8 > = len ) return ;
2018-12-21 21:00:03 +01:00
uint32_t timestamp = bufbe32toh ( buf + offset ) ; offset + = 4 ; // published timestamp (seconds)
uint16_t expires = bufbe16toh ( buf + offset ) ; offset + = 2 ; // expires (seconds)
SetExpirationTime ( ( timestamp + expires ) * 1000LL ) ; // in milliseconds
2018-12-31 20:23:48 +01:00
uint16_t flags = bufbe16toh ( buf + offset ) ; offset + = 2 ; // flags
2019-01-16 00:56:26 +01:00
std : : unique_ptr < i2p : : crypto : : Verifier > transientVerifier ;
2019-01-01 23:00:37 +01:00
if ( flags & 0x0001 )
{
2019-01-16 00:56:26 +01:00
// transient key
2019-01-02 15:43:18 +01:00
if ( offset + 6 > = len ) return ;
const uint8_t * signedData = buf + offset ;
2019-01-16 00:56:26 +01:00
uint32_t expiresTimestamp = bufbe32toh ( buf + offset ) ; offset + = 4 ; // expires timestamp
if ( expiresTimestamp < i2p : : util : : GetSecondsSinceEpoch ( ) )
{
LogPrint ( eLogWarning , " LeaseSet2: transient key expired " ) ;
return ;
}
2019-01-01 23:00:37 +01:00
uint16_t keyType = bufbe16toh ( buf + offset ) ; offset + = 2 ;
2019-01-16 00:56:26 +01:00
transientVerifier . reset ( i2p : : data : : IdentityEx : : CreateVerifier ( keyType ) ) ;
if ( ! transientVerifier ) return ;
auto keyLen = transientVerifier - > GetPublicKeyLen ( ) ;
2019-01-01 23:00:37 +01:00
if ( offset + keyLen > = len ) return ;
2019-01-16 00:56:26 +01:00
transientVerifier - > SetPublicKey ( buf + offset ) ; offset + = keyLen ;
2019-01-08 17:26:50 +01:00
if ( offset + identity - > GetSignatureLen ( ) > = len ) return ;
2019-01-02 15:43:18 +01:00
if ( ! identity - > Verify ( signedData , keyLen + 6 , buf + offset ) ) return ;
2019-01-08 17:26:50 +01:00
offset + = identity - > GetSignatureLen ( ) ;
2019-01-01 23:00:37 +01:00
}
2019-01-02 20:19:10 +01:00
// type specific part
size_t s = 0 ;
switch ( m_StoreType )
{
case NETDB_STORE_TYPE_STANDARD_LEASESET2 :
s = ReadStandardLS2TypeSpecificPart ( buf + offset , len - offset ) ;
break ;
case NETDB_STORE_TYPE_META_LEASESET2 :
s = ReadMetaLS2TypeSpecificPart ( buf + offset , len - offset ) ;
break ;
default :
LogPrint ( eLogWarning , " LeaseSet2: Unexpected store type " , ( int ) m_StoreType ) ;
}
if ( ! s ) return ;
offset + = s ;
// verify signature
2019-01-16 00:56:26 +01:00
bool verified = transientVerifier ? VerifySignature ( transientVerifier , buf , len , offset ) :
2019-01-08 17:26:50 +01:00
VerifySignature ( identity , buf , len , offset ) ;
SetIsValid ( verified ) ;
}
template < typename Verifier >
bool LeaseSet2 : : VerifySignature ( Verifier & verifier , const uint8_t * buf , size_t len , size_t signatureOffset )
{
if ( signatureOffset + verifier - > GetSignatureLen ( ) > len ) return false ;
2019-01-10 16:57:57 +01:00
// we assume buf inside DatabaseStore message, so buf[-1] is valid memory
// change it for signature verification, and restore back
uint8_t c = buf [ - 1 ] ;
const_cast < uint8_t * > ( buf ) [ - 1 ] = m_StoreType ;
bool verified = verifier - > Verify ( buf - 1 , signatureOffset + 1 , buf + signatureOffset ) ;
const_cast < uint8_t * > ( buf ) [ - 1 ] = c ;
2019-01-02 20:19:10 +01:00
if ( ! verified )
LogPrint ( eLogWarning , " LeaseSet2: verification failed " ) ;
2019-01-08 17:26:50 +01:00
return verified ;
2019-01-02 20:19:10 +01:00
}
size_t LeaseSet2 : : ReadStandardLS2TypeSpecificPart ( const uint8_t * buf , size_t len )
{
size_t offset = 0 ;
2018-12-26 21:27:32 +01:00
// properties
uint16_t propertiesLen = bufbe16toh ( buf + offset ) ; offset + = 2 ;
offset + = propertiesLen ; // skip for now. TODO: implement properties
2019-01-02 20:19:10 +01:00
if ( offset + 1 > = len ) return 0 ;
2018-12-26 21:27:32 +01:00
// key sections
2019-01-01 23:00:37 +01:00
int numKeySections = buf [ offset ] ; offset + + ;
for ( int i = 0 ; i < numKeySections ; i + + )
2018-12-26 21:27:32 +01:00
{
2019-01-09 18:47:47 +01:00
uint16_t keyType = bufbe16toh ( buf + offset ) ; offset + = 2 ; // encryption key type
2019-01-02 20:19:10 +01:00
if ( offset + 2 > = len ) return 0 ;
2018-12-26 21:27:32 +01:00
uint16_t encryptionKeyLen = bufbe16toh ( buf + offset ) ; offset + = 2 ;
2019-01-09 18:47:47 +01:00
if ( offset + encryptionKeyLen > = len ) return 0 ;
if ( ! m_Encryptor & & IsStoreLeases ( ) ) // create encryptor with leases only, first key
{
auto encryptor = i2p : : data : : IdentityEx : : CreateEncryptor ( keyType , buf + offset ) ;
m_Encryptor = encryptor ; // TODO: atomic
}
2018-12-26 21:27:32 +01:00
offset + = encryptionKeyLen ;
}
// leases
2019-01-02 20:19:10 +01:00
if ( offset + 1 > = len ) return 0 ;
2018-12-26 21:27:32 +01:00
int numLeases = buf [ offset ] ; offset + + ;
2019-01-02 21:40:48 +01:00
auto ts = i2p : : util : : GetMillisecondsSinceEpoch ( ) ;
2019-01-09 18:47:47 +01:00
if ( IsStoreLeases ( ) )
2019-01-02 20:19:10 +01:00
{
2019-01-09 18:47:47 +01:00
UpdateLeasesBegin ( ) ;
for ( int i = 0 ; i < numLeases ; i + + )
{
2019-01-10 17:52:34 +01:00
if ( offset + LEASE2_SIZE > len ) return 0 ;
2019-01-09 18:47:47 +01:00
Lease lease ;
lease . tunnelGateway = buf + offset ; offset + = 32 ; // gateway
lease . tunnelID = bufbe32toh ( buf + offset ) ; offset + = 4 ; // tunnel ID
lease . endDate = bufbe32toh ( buf + offset ) * 1000LL ; offset + = 4 ; // end date
UpdateLease ( lease , ts ) ;
}
UpdateLeasesEnd ( ) ;
2019-01-02 20:19:10 +01:00
}
2019-01-09 18:47:47 +01:00
else
2019-01-10 17:52:34 +01:00
offset + = numLeases * LEASE2_SIZE ; // 40 bytes per lease
2019-01-02 20:19:10 +01:00
return offset ;
}
size_t LeaseSet2 : : ReadMetaLS2TypeSpecificPart ( const uint8_t * buf , size_t len )
{
size_t offset = 0 ;
// properties
uint16_t propertiesLen = bufbe16toh ( buf + offset ) ; offset + = 2 ;
offset + = propertiesLen ; // skip for now. TODO: implement properties
// entries
if ( offset + 1 > = len ) return 0 ;
int numEntries = buf [ offset ] ; offset + + ;
for ( int i = 0 ; i < numEntries ; i + + )
{
if ( offset + 40 > = len ) return 0 ;
offset + = 32 ; // hash
offset + = 3 ; // flags
offset + = 1 ; // cost
offset + = 4 ; // expires
}
// revocations
if ( offset + 1 > = len ) return 0 ;
int numRevocations = buf [ offset ] ; offset + + ;
for ( int i = 0 ; i < numRevocations ; i + + )
{
if ( offset + 32 > len ) return 0 ;
offset + = 32 ; // hash
}
return offset ;
2018-12-21 21:00:03 +01:00
}
2019-01-08 17:26:50 +01:00
void LeaseSet2 : : ReadFromBufferEncrypted ( const uint8_t * buf , size_t len )
{
size_t offset = 0 ;
// blinded key
2019-01-14 19:49:27 +01:00
if ( len < 2 ) return ;
2019-01-08 17:26:50 +01:00
uint16_t blindedKeyType = bufbe16toh ( buf + offset ) ; offset + = 2 ;
std : : unique_ptr < i2p : : crypto : : Verifier > blindedVerifier ( i2p : : data : : IdentityEx : : CreateVerifier ( blindedKeyType ) ) ;
if ( ! blindedVerifier ) return ;
auto blindedKeyLen = blindedVerifier - > GetPublicKeyLen ( ) ;
if ( offset + blindedKeyLen > = len ) return ;
blindedVerifier - > SetPublicKey ( buf + offset ) ; offset + = blindedKeyLen ;
// expiration
if ( offset + 8 > = len ) return ;
uint32_t timestamp = bufbe32toh ( buf + offset ) ; offset + = 4 ; // published timestamp (seconds)
uint16_t expires = bufbe16toh ( buf + offset ) ; offset + = 2 ; // expires (seconds)
SetExpirationTime ( ( timestamp + expires ) * 1000LL ) ; // in milliseconds
uint16_t flags = bufbe16toh ( buf + offset ) ; offset + = 2 ; // flags
2019-01-16 00:56:26 +01:00
std : : unique_ptr < i2p : : crypto : : Verifier > transientVerifier ;
2019-01-08 17:26:50 +01:00
if ( flags & 0x0001 )
{
2019-01-16 00:56:26 +01:00
// transient key
2019-01-08 17:26:50 +01:00
if ( offset + 6 > = len ) return ;
const uint8_t * signedData = buf + offset ;
2019-01-16 00:56:26 +01:00
uint32_t expiresTimestamp = bufbe32toh ( buf + offset ) ; offset + = 4 ; // expires timestamp
if ( expiresTimestamp < i2p : : util : : GetSecondsSinceEpoch ( ) )
{
LogPrint ( eLogWarning , " LeaseSet2: transient key expired " ) ;
return ;
}
2019-01-08 17:26:50 +01:00
uint16_t keyType = bufbe16toh ( buf + offset ) ; offset + = 2 ;
2019-01-16 00:56:26 +01:00
transientVerifier . reset ( i2p : : data : : IdentityEx : : CreateVerifier ( keyType ) ) ;
if ( ! transientVerifier ) return ;
auto keyLen = transientVerifier - > GetPublicKeyLen ( ) ;
2019-01-08 17:26:50 +01:00
if ( offset + keyLen > = len ) return ;
2019-01-16 00:56:26 +01:00
transientVerifier - > SetPublicKey ( buf + offset ) ; offset + = keyLen ;
2019-01-08 17:26:50 +01:00
if ( offset + blindedVerifier - > GetSignatureLen ( ) > = len ) return ;
if ( ! blindedVerifier - > Verify ( signedData , keyLen + 6 , buf + offset ) ) return ;
offset + = blindedVerifier - > GetSignatureLen ( ) ;
}
// outer ciphertext
if ( offset + 2 > len ) return ;
uint16_t lenOuterCiphertext = bufbe16toh ( buf + offset ) ; offset + = 2 + lenOuterCiphertext ;
// verify signature
2019-01-16 00:56:26 +01:00
bool verified = transientVerifier ? VerifySignature ( transientVerifier , buf , len , offset ) :
2019-01-08 17:26:50 +01:00
VerifySignature ( blindedVerifier , buf , len , offset ) ;
SetIsValid ( verified ) ;
}
2019-01-09 18:47:47 +01:00
void LeaseSet2 : : Encrypt ( const uint8_t * data , uint8_t * encrypted , BN_CTX * ctx ) const
{
auto encryptor = m_Encryptor ; // TODO: atomic
if ( encryptor )
encryptor - > Encrypt ( data , encrypted , ctx , true ) ;
}
2017-11-07 21:05:22 +01:00
2019-01-14 19:49:27 +01:00
uint64_t LeaseSet2 : : ExtractTimestamp ( const uint8_t * buf , size_t len ) const
{
if ( len < 8 ) return 0 ;
if ( m_StoreType = = NETDB_STORE_TYPE_ENCRYPTED_LEASESET2 )
{
// encrypted LS2
size_t offset = 0 ;
uint16_t blindedKeyType = bufbe16toh ( buf + offset ) ; offset + = 2 ;
std : : unique_ptr < i2p : : crypto : : Verifier > blindedVerifier ( i2p : : data : : IdentityEx : : CreateVerifier ( blindedKeyType ) ) ;
if ( ! blindedVerifier ) return 0 ;
auto blindedKeyLen = blindedVerifier - > GetPublicKeyLen ( ) ;
if ( offset + blindedKeyLen + 6 > = len ) return 0 ;
offset + = blindedKeyLen ;
uint32_t timestamp = bufbe32toh ( buf + offset ) ; offset + = 4 ;
uint16_t expires = bufbe16toh ( buf + offset ) ; offset + = 2 ;
return ( timestamp + expires ) * 1000LL ;
}
else
{
auto identity = GetIdentity ( ) ;
if ( ! identity ) return 0 ;
size_t offset = identity - > GetFullLen ( ) ;
if ( offset + 6 > = len ) return 0 ;
uint32_t timestamp = bufbe32toh ( buf + offset ) ; offset + = 4 ;
uint16_t expires = bufbe16toh ( buf + offset ) ; offset + = 2 ;
return ( timestamp + expires ) * 1000LL ;
}
}
2016-05-25 20:17:34 +02:00
LocalLeaseSet : : LocalLeaseSet ( std : : shared_ptr < const IdentityEx > identity , const uint8_t * encryptionPublicKey , std : : vector < std : : shared_ptr < i2p : : tunnel : : InboundTunnel > > tunnels ) :
2016-05-25 21:10:28 +02:00
m_ExpirationTime ( 0 ) , m_Identity ( identity )
2016-05-25 20:17:34 +02:00
{
int num = tunnels . size ( ) ;
if ( num > MAX_NUM_LEASES ) num = MAX_NUM_LEASES ;
// identity
2016-05-25 23:41:24 +02:00
auto signingKeyLen = m_Identity - > GetSigningPublicKeyLen ( ) ;
2018-01-06 04:48:51 +01:00
m_BufferLen = m_Identity - > GetFullLen ( ) + 256 + signingKeyLen + 1 + num * LEASE_SIZE + m_Identity - > GetSignatureLen ( ) ;
m_Buffer = new uint8_t [ m_BufferLen ] ;
2016-05-25 20:17:34 +02:00
auto offset = m_Identity - > ToBuffer ( m_Buffer , m_BufferLen ) ;
memcpy ( m_Buffer + offset , encryptionPublicKey , 256 ) ;
offset + = 256 ;
memset ( m_Buffer + offset , 0 , signingKeyLen ) ;
offset + = signingKeyLen ;
// num leases
2018-01-06 04:48:51 +01:00
m_Buffer [ offset ] = num ;
2016-05-25 20:17:34 +02:00
offset + + ;
// leases
2016-05-29 22:35:57 +02:00
m_Leases = m_Buffer + offset ;
2016-05-25 20:17:34 +02:00
auto currentTime = i2p : : util : : GetMillisecondsSinceEpoch ( ) ;
for ( int i = 0 ; i < num ; i + + )
{
memcpy ( m_Buffer + offset , tunnels [ i ] - > GetNextIdentHash ( ) , 32 ) ;
offset + = 32 ; // gateway id
htobe32buf ( m_Buffer + offset , tunnels [ i ] - > GetNextTunnelID ( ) ) ;
offset + = 4 ; // tunnel id
uint64_t ts = tunnels [ i ] - > GetCreationTime ( ) + i2p : : tunnel : : TUNNEL_EXPIRATION_TIMEOUT - i2p : : tunnel : : TUNNEL_EXPIRATION_THRESHOLD ; // 1 minute before expiration
ts * = 1000 ; // in milliseconds
2016-05-25 21:10:28 +02:00
if ( ts > m_ExpirationTime ) m_ExpirationTime = ts ;
2016-05-25 20:17:34 +02:00
// make sure leaseset is newer than previous, but adding some time to expiration date
ts + = ( currentTime - tunnels [ i ] - > GetCreationTime ( ) * 1000LL ) * 2 / i2p : : tunnel : : TUNNEL_EXPIRATION_TIMEOUT ; // up to 2 secs
htobe64buf ( m_Buffer + offset , ts ) ;
offset + = 8 ; // end date
}
// we don't sign it yet. must be signed later on
2016-05-25 21:10:28 +02:00
}
2016-05-25 20:17:34 +02:00
2016-05-30 18:56:42 +02:00
LocalLeaseSet : : LocalLeaseSet ( std : : shared_ptr < const IdentityEx > identity , const uint8_t * buf , size_t len ) :
m_ExpirationTime ( 0 ) , m_Identity ( identity )
{
2019-01-09 20:51:47 +01:00
if ( buf )
{
m_BufferLen = len ;
m_Buffer = new uint8_t [ m_BufferLen ] ;
memcpy ( m_Buffer , buf , len ) ;
}
else
{
m_Buffer = nullptr ;
m_BufferLen = 0 ;
}
2016-05-30 18:56:42 +02:00
}
2016-05-25 21:10:28 +02:00
bool LocalLeaseSet : : IsExpired ( ) const
2016-05-25 20:17:34 +02:00
{
2016-05-25 21:10:28 +02:00
auto ts = i2p : : util : : GetMillisecondsSinceEpoch ( ) ;
return ts > m_ExpirationTime ;
2018-01-06 04:48:51 +01:00
}
2018-01-24 16:16:51 +01:00
bool LeaseSetBufferValidate ( const uint8_t * ptr , size_t sz , uint64_t & expires )
{
IdentityEx ident ( ptr , sz ) ;
size_t size = ident . GetFullLen ( ) ;
if ( size > sz )
{
LogPrint ( eLogError , " LeaseSet: identity length " , size , " exceeds buffer size " , sz ) ;
return false ;
}
// encryption key
size + = 256 ;
// signing key (unused)
size + = ident . GetSigningPublicKeyLen ( ) ;
uint8_t numLeases = ptr [ size ] ;
+ + size ;
if ( ! numLeases | | numLeases > MAX_NUM_LEASES )
{
LogPrint ( eLogError , " LeaseSet: incorrect number of leases " , ( int ) numLeases ) ;
return false ;
}
const uint8_t * leases = ptr + size ;
expires = 0 ;
/** find lease with the max expiration timestamp */
for ( int i = 0 ; i < numLeases ; i + + )
{
leases + = 36 ; // gateway + tunnel ID
uint64_t endDate = bufbe64toh ( leases ) ;
leases + = 8 ; // end date
if ( endDate > expires )
expires = endDate ;
}
return ident . Verify ( ptr , leases - ptr , leases ) ;
}
2019-01-09 20:51:47 +01:00
LocalLeaseSet2 : : LocalLeaseSet2 ( uint8_t storeType , std : : shared_ptr < const IdentityEx > identity ,
uint16_t keyType , uint16_t keyLen , const uint8_t * encryptionPublicKey ,
std : : vector < std : : shared_ptr < i2p : : tunnel : : InboundTunnel > > tunnels ) :
2019-01-10 17:52:34 +01:00
LocalLeaseSet ( identity , nullptr , 0 )
2019-01-09 20:51:47 +01:00
{
2019-01-10 17:52:34 +01:00
// assume standard LS2
int num = tunnels . size ( ) ;
if ( num > MAX_NUM_LEASES ) num = MAX_NUM_LEASES ;
m_BufferLen = identity - > GetFullLen ( ) + 4 /*published*/ + 2 /*expires*/ + 2 /*flag*/ + 2 /*properties len*/ +
1 /*num keys*/ + 2 /*key type*/ + 2 /*key len*/ + keyLen /*key*/ + 1 /*num leases*/ + num * LEASE2_SIZE + identity - > GetSignatureLen ( ) ;
m_Buffer = new uint8_t [ m_BufferLen + 1 ] ;
m_Buffer [ 0 ] = storeType ;
// LS2 header
auto offset = identity - > ToBuffer ( m_Buffer + 1 , m_BufferLen ) + 1 ;
auto timestamp = i2p : : util : : GetSecondsSinceEpoch ( ) ;
htobe32buf ( m_Buffer + offset , timestamp ) ; offset + = 4 ; // published timestamp (seconds)
uint8_t * expiresBuf = m_Buffer + offset ; offset + = 2 ; // expires, fill later
htobe16buf ( m_Buffer + offset , 0 ) ; offset + = 2 ; // flags
htobe16buf ( m_Buffer + offset , 0 ) ; offset + = 2 ; // properties len
// keys
m_Buffer [ offset ] = 1 ; offset + + ; // 1 key
htobe16buf ( m_Buffer + offset , keyType ) ; offset + = 2 ; // key type
htobe16buf ( m_Buffer + offset , keyLen ) ; offset + = 2 ; // key len
memcpy ( m_Buffer + offset , encryptionPublicKey , keyLen ) ; offset + = keyLen ; // key
// leases
uint32_t expirationTime = 0 ; // in seconds
m_Buffer [ offset ] = num ; offset + + ; // num leases
for ( int i = 0 ; i < num ; i + + )
{
memcpy ( m_Buffer + offset , tunnels [ i ] - > GetNextIdentHash ( ) , 32 ) ;
offset + = 32 ; // gateway id
htobe32buf ( m_Buffer + offset , tunnels [ i ] - > GetNextTunnelID ( ) ) ;
offset + = 4 ; // tunnel id
auto ts = tunnels [ i ] - > GetCreationTime ( ) + i2p : : tunnel : : TUNNEL_EXPIRATION_TIMEOUT - i2p : : tunnel : : TUNNEL_EXPIRATION_THRESHOLD ; // in seconds, 1 minute before expiration
if ( ts > expirationTime ) expirationTime = ts ;
htobe32buf ( m_Buffer + offset , ts ) ;
offset + = 4 ; // end date
}
// update expiration
SetExpirationTime ( expirationTime * 1000LL ) ;
auto expires = expirationTime - timestamp ;
htobe16buf ( expiresBuf , expires > 0 ? expires : 0 ) ;
// we don't sign it yet. must be signed later on
2019-01-09 20:51:47 +01:00
}
2019-01-29 17:30:31 +01:00
LocalLeaseSet2 : : LocalLeaseSet2 ( uint8_t storeType , std : : shared_ptr < const IdentityEx > identity , const uint8_t * buf , size_t len ) :
LocalLeaseSet ( identity , nullptr , 0 )
{
m_BufferLen = len ;
m_Buffer = new uint8_t [ m_BufferLen + 1 ] ;
memcpy ( m_Buffer + 1 , buf , len ) ;
m_Buffer [ 0 ] = storeType ;
}
2018-01-06 04:48:51 +01:00
}
}