2022-02-27 07:53:44 -05:00
# include <STDInclude.hpp>
2023-01-03 07:16:44 -05:00
# include <Utils/InfoString.hpp>
# include <proto/auth.pb.h>
2022-12-26 07:07:24 -05:00
# include "Bans.hpp"
2017-01-19 16:23:59 -05:00
namespace Components
{
Auth : : TokenIncrementing Auth : : TokenContainer ;
Utils : : Cryptography : : Token Auth : : GuidToken ;
Utils : : Cryptography : : Token Auth : : ComputeToken ;
Utils : : Cryptography : : ECC : : Key Auth : : GuidKey ;
2022-02-26 17:50:53 -05:00
std : : vector < std : : uint64_t > Auth : : BannedUids =
{
2020-08-08 06:19:55 -04:00
0xf4d2c30b712ac6e3 ,
0xf7e33c4081337fa3 ,
0x6f5597f103cc50e9
2019-10-03 03:10:00 -04:00
} ;
2017-01-19 16:23:59 -05:00
void Auth : : Frame ( )
{
if ( Auth : : TokenContainer . generating )
{
static double mseconds = 0 ;
static Utils : : Time : : Interval interval ;
if ( interval . elapsed ( 500 ms ) )
{
interval . update ( ) ;
int diff = Game : : Sys_Milliseconds ( ) - Auth : : TokenContainer . startTime ;
double hashPMS = ( Auth : : TokenContainer . hashes * 1.0 ) / diff ;
double requiredHashes = std : : pow ( 2 , Auth : : TokenContainer . targetLevel + 1 ) - Auth : : TokenContainer . hashes ;
mseconds = requiredHashes / hashPMS ;
if ( mseconds < 0 ) mseconds = 0 ;
}
Localization : : Set ( " MPUI_SECURITY_INCREASE_MESSAGE " , Utils : : String : : VA ( " Increasing security level from %d to %d (est. %s) " , Auth : : GetSecurityLevel ( ) , Auth : : TokenContainer . targetLevel , Utils : : String : : FormatTimeSpan ( static_cast < int > ( mseconds ) ) . data ( ) ) ) ;
}
else if ( Auth : : TokenContainer . thread . joinable ( ) )
{
Auth : : TokenContainer . thread . join ( ) ;
Auth : : TokenContainer . generating = false ;
Auth : : StoreKey ( ) ;
2022-06-22 04:58:51 -04:00
Logger : : Debug ( " Security level is {} " , Auth : : GetSecurityLevel ( ) ) ;
2017-01-19 16:23:59 -05:00
Command : : Execute ( " closemenu security_increase_popmenu " , false ) ;
if ( ! Auth : : TokenContainer . cancel )
{
if ( Auth : : TokenContainer . command . empty ( ) )
{
2017-02-06 15:09:41 -05:00
Game : : ShowMessageBox ( Utils : : String : : VA ( " Your new security level is %d " , Auth : : GetSecurityLevel ( ) ) , " Success " ) ;
2017-01-19 16:23:59 -05:00
}
else
{
Toast : : Show ( " cardicon_locked " , " Success " , Utils : : String : : VA ( " Your new security level is %d " , Auth : : GetSecurityLevel ( ) ) , 5000 ) ;
Command : : Execute ( Auth : : TokenContainer . command , false ) ;
}
}
Auth : : TokenContainer . cancel = false ;
}
}
2019-10-03 03:10:00 -04:00
2023-04-06 11:28:57 -04:00
void Auth : : SendConnectDataStub ( Game : : netsrc_t sock , Game : : netadr_t adr , const char * format , int len )
2017-01-19 16:23:59 -05:00
{
// Ensure our certificate is loaded
Steam : : SteamUser ( ) - > GetSteamID ( ) ;
if ( ! Auth : : GuidKey . isValid ( ) )
{
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_SERVERDISCONNECT , " Connecting failed: Guid key is invalid! " ) ;
2017-01-19 16:23:59 -05:00
return ;
}
2019-10-03 03:10:00 -04:00
if ( std : : find ( Auth : : BannedUids . begin ( ) , Auth : : BannedUids . end ( ) , Steam : : SteamUser ( ) - > GetSteamID ( ) . bits ) ! = Auth : : BannedUids . end ( ) )
{
2019-10-03 03:16:26 -04:00
Auth : : GenerateKey ( ) ;
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_SERVERDISCONNECT , " Your online profile is invalid. A new key has been generated. " ) ;
2019-10-03 03:10:00 -04:00
return ;
}
2017-01-19 16:23:59 -05:00
std : : string connectString ( format , len ) ;
Game : : SV_Cmd_TokenizeString ( connectString . data ( ) ) ;
Command : : ServerParams params ;
2022-03-17 14:50:20 -04:00
if ( params . size ( ) < 3 )
2017-01-19 16:23:59 -05:00
{
Game : : SV_Cmd_EndTokenizedString ( ) ;
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_SERVERDISCONNECT , " Connecting failed: Command parsing error! " ) ;
2017-01-19 16:23:59 -05:00
return ;
}
Utils : : InfoString infostr ( params [ 2 ] ) ;
std : : string challenge = infostr . get ( " challenge " ) ;
if ( challenge . empty ( ) )
{
Game : : SV_Cmd_EndTokenizedString ( ) ;
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_SERVERDISCONNECT , " Connecting failed: Challenge parsing error! " ) ;
2017-01-19 16:23:59 -05:00
return ;
}
2019-01-10 15:18:18 -05:00
if ( Steam : : Enabled ( ) & & ! Friends : : IsInvisible ( ) & & ! Dvar : : Var ( " cl_anonymous " ) . get < bool > ( ) & & Steam : : Proxy : : SteamUser_ )
2017-02-17 06:26:07 -05:00
{
2017-02-18 03:42:55 -05:00
infostr . set ( " realsteamId " , Utils : : String : : VA ( " %llX " , Steam : : Proxy : : SteamUser_ - > GetSteamID ( ) . bits ) ) ;
2017-02-17 06:26:07 -05:00
}
// Build new connect string
connectString . clear ( ) ;
connectString . append ( params [ 0 ] ) ;
connectString . append ( " " ) ;
connectString . append ( params [ 1 ] ) ;
connectString . append ( " " ) ;
connectString . append ( " \" " + infostr . build ( ) + " \" " ) ;
2017-01-19 16:23:59 -05:00
Game : : SV_Cmd_EndTokenizedString ( ) ;
Proto : : Auth : : Connect connectData ;
connectData . set_token ( Auth : : GuidToken . toString ( ) ) ;
connectData . set_publickey ( Auth : : GuidKey . getPublicKey ( ) ) ;
connectData . set_signature ( Utils : : Cryptography : : ECC : : SignMessage ( Auth : : GuidKey , challenge ) ) ;
connectData . set_infostring ( connectString ) ;
Network : : SendCommand ( sock , adr , " connect " , connectData . SerializeAsString ( ) ) ;
}
2017-02-01 07:44:25 -05:00
void Auth : : ParseConnectData ( Game : : msg_t * msg , Game : : netadr_t * addr )
2017-01-19 16:23:59 -05:00
{
Network : : Address address ( addr ) ;
// Parse proto data
Proto : : Auth : : Connect connectData ;
2022-08-13 11:19:45 -04:00
if ( msg - > cursize < = 12 | | ! connectData . ParseFromString ( std : : string ( reinterpret_cast < char * > ( & msg - > data [ 12 ] ) , msg - > cursize - 12 ) ) )
2017-01-19 16:23:59 -05:00
{
Network : : Send ( address , " error \n Invalid connect packet! " ) ;
return ;
}
2017-01-20 08:36:52 -05:00
// Simply connect, if we're in debug mode, we ignore all security checks
# ifndef DEBUG
if ( address . isLoopback ( ) )
2017-01-19 16:23:59 -05:00
# endif
{
if ( ! connectData . infostring ( ) . empty ( ) )
{
Game : : SV_Cmd_EndTokenizedString ( ) ;
Game : : SV_Cmd_TokenizeString ( connectData . infostring ( ) . data ( ) ) ;
Game : : SV_DirectConnect ( * address . get ( ) ) ;
}
else
{
Network : : Send ( address , " error \n Invalid infostring data! " ) ;
}
}
2017-01-20 08:36:52 -05:00
# ifndef DEBUG
2017-01-19 16:23:59 -05:00
else
{
// Validate proto data
if ( connectData . signature ( ) . empty ( ) | | connectData . publickey ( ) . empty ( ) | | connectData . token ( ) . empty ( ) | | connectData . infostring ( ) . empty ( ) )
{
Network : : Send ( address , " error \n Invalid connect data! " ) ;
return ;
}
// Setup new cmd params
Game : : SV_Cmd_EndTokenizedString ( ) ;
Game : : SV_Cmd_TokenizeString ( connectData . infostring ( ) . data ( ) ) ;
// Access the params
Command : : ServerParams params ;
// Ensure there are enough params
2022-03-17 14:50:20 -04:00
if ( params . size ( ) < 3 )
2017-01-19 16:23:59 -05:00
{
Network : : Send ( address , " error \n Invalid connect string! " ) ;
return ;
}
// Parse the infostring
Utils : : InfoString infostr ( params [ 2 ] ) ;
// Read the required data
2023-04-06 11:28:57 -04:00
const auto steamId = infostr . get ( " xuid " ) ;
const auto challenge = infostr . get ( " challenge " ) ;
2017-01-19 16:23:59 -05:00
if ( steamId . empty ( ) | | challenge . empty ( ) )
{
Network : : Send ( address , " error \n Invalid connect data! " ) ;
return ;
}
// Parse the id
2022-02-26 18:02:04 -05:00
const auto xuid = std : : strtoull ( steamId . data ( ) , nullptr , 16 ) ;
2017-01-19 16:23:59 -05:00
SteamID guid ;
2017-02-18 03:42:55 -05:00
guid . bits = xuid ;
2017-01-19 16:23:59 -05:00
2022-02-26 17:50:53 -05:00
if ( Bans : : IsBanned ( { guid , address . getIP ( ) } ) )
2017-01-19 16:23:59 -05:00
{
Network : : Send ( address , " error \n EXE_ERR_BANNED_PERM " ) ;
return ;
}
2019-10-03 03:10:00 -04:00
if ( std : : find ( Auth : : BannedUids . begin ( ) , Auth : : BannedUids . end ( ) , xuid ) ! = Auth : : BannedUids . end ( ) )
{
Network : : Send ( address , " error \n Your online profile is invalid. Delete your players folder and restart ^2IW4x^7. " ) ;
return ;
}
2017-01-19 16:23:59 -05:00
if ( xuid ! = Auth : : GetKeyHash ( connectData . publickey ( ) ) )
{
Network : : Send ( address , " error \n XUID doesn't match the certificate! " ) ;
return ;
}
// Verify the signature
Utils : : Cryptography : : ECC : : Key key ;
key . set ( connectData . publickey ( ) ) ;
if ( ! key . isValid ( ) | | ! Utils : : Cryptography : : ECC : : VerifyMessage ( key , challenge , connectData . signature ( ) ) )
{
Network : : Send ( address , " error \n Challenge signature was invalid! " ) ;
return ;
}
// Verify the security level
2022-02-26 17:50:53 -05:00
auto ourLevel = Dvar : : Var ( " sv_securityLevel " ) . get < unsigned int > ( ) ;
auto userLevel = Auth : : GetZeroBits ( connectData . token ( ) , connectData . publickey ( ) ) ;
2017-01-19 16:23:59 -05:00
if ( userLevel < ourLevel )
{
Network : : Send ( address , Utils : : String : : VA ( " error \n Your security level (%d) is lower than the server's security level (%d) " , userLevel , ourLevel ) ) ;
return ;
}
2022-08-19 19:10:35 -04:00
Logger : : Debug ( " Verified XUID {:#X} ({}) from {} " , xuid , userLevel , address . getString ( ) ) ;
2017-01-19 16:23:59 -05:00
Game : : SV_DirectConnect ( * address . get ( ) ) ;
}
2017-01-20 08:36:52 -05:00
# endif
2017-01-19 16:23:59 -05:00
}
__declspec ( naked ) void Auth : : DirectConnectStub ( )
{
__asm
{
2017-02-01 07:44:25 -05:00
pushad
lea eax , [ esp + 20 h ]
push eax
2017-01-19 16:23:59 -05:00
push esi
call Auth : : ParseConnectData
pop esi
2017-02-01 07:44:25 -05:00
pop eax
popad
2017-01-19 16:23:59 -05:00
2017-02-01 07:44:25 -05:00
push 6265F Eh
retn
2017-01-19 16:23:59 -05:00
}
}
2018-12-17 08:29:18 -05:00
unsigned __int64 Auth : : GetKeyHash ( const std : : string & key )
2017-01-19 16:23:59 -05:00
{
std : : string hash = Utils : : Cryptography : : SHA1 : : Compute ( key ) ;
if ( hash . size ( ) > = 8 )
{
return * reinterpret_cast < unsigned __int64 * > ( const_cast < char * > ( hash . data ( ) ) ) ;
}
return 0 ;
}
unsigned __int64 Auth : : GetKeyHash ( )
{
Auth : : LoadKey ( ) ;
return Auth : : GetKeyHash ( Auth : : GuidKey . getPublicKey ( ) ) ;
}
void Auth : : StoreKey ( )
{
2022-10-15 16:31:16 -04:00
if ( ! Dedicated : : IsEnabled ( ) & & ! ZoneBuilder : : IsEnabled ( ) & & Auth : : GuidKey . isValid ( ) )
2017-01-19 16:23:59 -05:00
{
Proto : : Auth : : Certificate cert ;
cert . set_token ( Auth : : GuidToken . toString ( ) ) ;
cert . set_ctoken ( Auth : : ComputeToken . toString ( ) ) ;
cert . set_privatekey ( Auth : : GuidKey . serialize ( PK_PRIVATE ) ) ;
Utils : : IO : : WriteFile ( " players/guid.dat " , cert . SerializeAsString ( ) ) ;
}
}
2019-10-03 03:16:26 -04:00
void Auth : : GenerateKey ( )
{
Auth : : GuidToken . clear ( ) ;
Auth : : ComputeToken . clear ( ) ;
Auth : : GuidKey = Utils : : Cryptography : : ECC : : GenerateKey ( 512 ) ;
Auth : : StoreKey ( ) ;
}
2017-01-19 16:23:59 -05:00
void Auth : : LoadKey ( bool force )
{
if ( Dedicated : : IsEnabled ( ) | | ZoneBuilder : : IsEnabled ( ) ) return ;
if ( ! force & & Auth : : GuidKey . isValid ( ) ) return ;
Proto : : Auth : : Certificate cert ;
if ( cert . ParseFromString ( : : Utils : : IO : : ReadFile ( " players/guid.dat " ) ) )
{
Auth : : GuidKey . deserialize ( cert . privatekey ( ) ) ;
Auth : : GuidToken = cert . token ( ) ;
Auth : : ComputeToken = cert . ctoken ( ) ;
}
else
{
Auth : : GuidKey . free ( ) ;
}
if ( ! Auth : : GuidKey . isValid ( ) )
{
2019-10-03 03:16:26 -04:00
Auth : : GenerateKey ( ) ;
2017-01-19 16:23:59 -05:00
}
}
uint32_t Auth : : GetSecurityLevel ( )
{
return Auth : : GetZeroBits ( Auth : : GuidToken , Auth : : GuidKey . getPublicKey ( ) ) ;
}
2018-12-17 08:29:18 -05:00
void Auth : : IncreaseSecurityLevel ( uint32_t level , const std : : string & command )
2017-01-19 16:23:59 -05:00
{
if ( Auth : : GetSecurityLevel ( ) > = level ) return ;
if ( ! Auth : : TokenContainer . generating )
{
Auth : : TokenContainer . cancel = false ;
Auth : : TokenContainer . targetLevel = level ;
Auth : : TokenContainer . command = command ;
// Open menu
Command : : Execute ( " openmenu security_increase_popmenu " , true ) ;
// Start thread
2017-04-28 18:18:51 -04:00
Auth : : TokenContainer . thread = std : : thread ( [ & level ] ( )
2017-01-19 16:23:59 -05:00
{
Auth : : TokenContainer . generating = true ;
Auth : : TokenContainer . hashes = 0 ;
Auth : : TokenContainer . startTime = Game : : Sys_Milliseconds ( ) ;
Auth : : IncrementToken ( Auth : : GuidToken , Auth : : ComputeToken , Auth : : GuidKey . getPublicKey ( ) , Auth : : TokenContainer . targetLevel , & Auth : : TokenContainer . cancel , & Auth : : TokenContainer . hashes ) ;
Auth : : TokenContainer . generating = false ;
if ( Auth : : TokenContainer . cancel )
{
Logger : : Print ( " Token incrementation thread terminated \n " ) ;
}
} ) ;
}
}
2018-12-17 08:29:18 -05:00
uint32_t Auth : : GetZeroBits ( Utils : : Cryptography : : Token token , const std : : string & publicKey )
2017-01-19 16:23:59 -05:00
{
std : : string message = publicKey + token . toString ( ) ;
std : : string hash = Utils : : Cryptography : : SHA512 : : Compute ( message , false ) ;
uint32_t bits = 0 ;
for ( unsigned int i = 0 ; i < hash . size ( ) ; + + i )
{
if ( hash [ i ] = = ' \0 ' )
{
bits + = 8 ;
continue ;
}
uint8_t value = static_cast < uint8_t > ( hash [ i ] ) ;
for ( int j = 7 ; j > = 0 ; - - j )
{
if ( ( value > > j ) & 1 )
{
return bits ;
}
+ + bits ;
}
}
return bits ;
}
2018-12-17 08:29:18 -05:00
void Auth : : IncrementToken ( Utils : : Cryptography : : Token & token , Utils : : Cryptography : : Token & computeToken , const std : : string & publicKey , uint32_t zeroBits , bool * cancel , uint64_t * count )
2017-01-19 16:23:59 -05:00
{
if ( zeroBits > 512 ) return ; // Not possible, due to SHA512
if ( computeToken < token )
{
computeToken = token ;
}
// Check if we already have the desired security level
uint32_t lastLevel = Auth : : GetZeroBits ( token , publicKey ) ;
uint32_t level = lastLevel ;
if ( level > = zeroBits ) return ;
do
{
+ + computeToken ;
if ( count ) + + ( * count ) ;
level = Auth : : GetZeroBits ( computeToken , publicKey ) ;
// Store level if higher than the last one
if ( level > = lastLevel )
{
token = computeToken ;
lastLevel = level ;
}
// Allow canceling that shit
if ( cancel & & * cancel ) return ;
2017-04-28 18:18:51 -04:00
} while ( level < zeroBits ) ;
2017-01-19 16:23:59 -05:00
token = computeToken ;
}
Auth : : Auth ( )
{
Auth : : TokenContainer . cancel = false ;
Auth : : TokenContainer . generating = false ;
Localization : : Set ( " MPUI_SECURITY_INCREASE_MESSAGE " , " " ) ;
// Load the key
Auth : : LoadKey ( true ) ;
Steam : : SteamUser ( ) - > GetSteamID ( ) ;
2022-05-05 10:03:14 -04:00
Scheduler : : Loop ( Auth : : Frame , Scheduler : : Pipeline : : MAIN ) ;
2017-01-19 16:23:59 -05:00
// Register dvar
2022-07-02 13:52:57 -04:00
Dvar : : Register < int > ( " sv_securityLevel " , 23 , 0 , 512 , Game : : DVAR_SERVERINFO , " Security level for GUID certificates (POW) " ) ;
2017-01-19 16:23:59 -05:00
// Install registration hook
Utils : : Hook ( 0x6265F9 , Auth : : DirectConnectStub , HOOK_JUMP ) . install ( ) - > quick ( ) ;
Utils : : Hook ( 0x41D3E3 , Auth : : SendConnectDataStub , HOOK_CALL ) . install ( ) - > quick ( ) ;
// SteamIDs can only contain 31 bits of actual 'id' data.
// The other 33 bits are steam internal data like universe and so on.
// Using only 31 bits for fingerprints is pretty insecure.
// The function below verifies the integrity steam's part of the SteamID.
// Patching that check allows us to use 64 bit for fingerprints.
Utils : : Hook : : Set < DWORD > ( 0x4D0D60 , 0xC301B0 ) ;
// Guid command
2023-03-23 10:08:21 -04:00
Command : : Add ( " guid " , [ ]
2017-01-19 16:23:59 -05:00
{
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Your guid: {:#X} \n " , Steam : : SteamUser ( ) - > GetSteamID ( ) . bits ) ;
2017-01-19 16:23:59 -05:00
} ) ;
if ( ! Dedicated : : IsEnabled ( ) & & ! ZoneBuilder : : IsEnabled ( ) )
{
2017-04-28 18:18:51 -04:00
Command : : Add ( " securityLevel " , [ ] ( Command : : Params * params )
2017-01-19 16:23:59 -05:00
{
2022-03-17 14:50:20 -04:00
if ( params - > size ( ) < 2 )
2017-01-19 16:23:59 -05:00
{
2022-06-12 18:02:20 -04:00
const auto level = Auth : : GetZeroBits ( Auth : : GuidToken , Auth : : GuidKey . getPublicKey ( ) ) ;
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Your current security level is {} \n " , level ) ;
Logger : : Print ( " Your security token is: {} \n " , Utils : : String : : DumpHex ( Auth : : GuidToken . toString ( ) , " " ) ) ;
Logger : : Print ( " Your computation token is: {} \n " , Utils : : String : : DumpHex ( Auth : : ComputeToken . toString ( ) , " " ) ) ;
2017-01-19 16:23:59 -05:00
Toast : : Show ( " cardicon_locked " , " ^5Security Level " , Utils : : String : : VA ( " Your security level is %d " , level ) , 3000 ) ;
}
else
{
2023-03-09 13:39:10 -05:00
const auto level = std : : strtoul ( params - > get ( 1 ) , nullptr , 10 ) ;
2017-01-19 16:23:59 -05:00
Auth : : IncreaseSecurityLevel ( level ) ;
}
} ) ;
}
2022-08-24 10:38:14 -04:00
UIScript : : Add ( " security_increase_cancel " , [ ] ( [[maybe_unused]] const UIScript::Token& token, [[maybe_unused]] const Game : : uiInfo_s * info )
2017-01-19 16:23:59 -05:00
{
Auth : : TokenContainer . cancel = true ;
Logger : : Print ( " Token incrementation process canceled! \n " ) ;
} ) ;
}
Auth : : ~ Auth ( )
2017-01-23 16:06:50 -05:00
{
Auth : : StoreKey ( ) ;
}
void Auth : : preDestroy ( )
2017-01-19 16:23:59 -05:00
{
Auth : : TokenContainer . cancel = true ;
Auth : : TokenContainer . generating = false ;
// Terminate thread
if ( Auth : : TokenContainer . thread . joinable ( ) )
{
Auth : : TokenContainer . thread . join ( ) ;
}
}
bool Auth : : unitTest ( )
{
bool success = true ;
printf ( " Testing logical token operators: \n " ) ;
Utils : : Cryptography : : Token token1 ;
Utils : : Cryptography : : Token token2 ;
+ + token1 , token2 + + ; // Test incrementation operator
printf ( " Operator == : " ) ;
if ( token1 = = token2 & & ! ( + + token1 = = token2 ) ) printf ( " Success \n " ) ;
else
{
printf ( " Error \n " ) ;
success = false ;
}
printf ( " Operator != : " ) ;
if ( token1 ! = token2 & & ! ( + + token2 ! = token1 ) ) printf ( " Success \n " ) ;
else
{
printf ( " Error \n " ) ;
success = false ;
}
printf ( " Operator >= : " ) ;
if ( token1 > = token2 & & + + token1 > = token2 ) printf ( " Success \n " ) ;
else
{
printf ( " Error \n " ) ;
success = false ;
}
printf ( " Operator > : " ) ;
if ( token1 > token2 ) printf ( " Success \n " ) ;
else
{
printf ( " Error \n " ) ;
success = false ;
}
printf ( " Operator <= : " ) ;
if ( token1 < = + + token2 & & token1 < = + + token2 ) printf ( " Success \n " ) ;
else
{
printf ( " Error \n " ) ;
success = false ;
}
printf ( " Operator < : " ) ;
if ( token1 < token2 ) printf ( " Success \n " ) ;
else
{
printf ( " Error \n " ) ;
success = false ;
}
return success ;
}
}