csgo-2018-source/game/client/cstrike15/cs_app_lifetime_gamestats.cpp
2021-07-24 21:11:47 -07:00

278 lines
7.5 KiB
C++

//========= Copyright © 1996-2005, Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
#include "cbase.h"
#include "cs_app_lifetime_gamestats.h"
#include "steamworks_gamestats.h"
#include "steam/isteamutils.h"
#include "csgo_serialnumbersequence.h"
#include "gametypes.h"
#include "matchmaking/imatchframework.h"
#include "cs_gamerules.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
extern IGameTypes *g_pGameTypes;
ConVar cl_debug_client_gamestats( "cl_debug_client_gamestats", "1" );
static CS_App_Lifetime_Gamestats g_CSAppLifetimeGamestats;
CS_App_Lifetime_Gamestats* CSAppLifetimeGameStats() { return &g_CSAppLifetimeGamestats; }
uint16 CS_App_Lifetime_Gamestats::m_unEventCount = 0;
const char* g_szValidUIEvents [] =
{
"ServerBrowserLAN",
"ServerBrowserFriends",
"ServerBrowserJoinFriend",
"ServerBrowserUnknownJoinType",
"ServerBrowserFavorites",
"ServerBrowserHistory",
"ServerBrowserSpectator",
"ServerBrowserInternet",
"ServerBrowserUnknownPageType",
"ServerBrowserJoinByIP",
"HLTVConnectRelay",
"CommunityQuickPlay",
"ConnectStringOnCommandline",
"PurchaseEventRequired",
"PurchaseEventMinutesPlayed",
"PurchaseEventNoMinutesPlayed",
"PurchaseEventStash",
"InventoryEquipMainMenu",
"InventoryEquipConnectedActive",
"InventoryEquipConnectedInactive",
"CrateUnlockMainMenu",
"CrateUnlockConnected",
"WeaponInspectMainMenuDefault",
"WeaponInspectMainMenuPainted",
"WeaponInspectConnectedDefault",
"WeaponInspectConnectedPainted",
"WorkshopWeaponPreview",
"MatchDownloadAttempt",
"WatchLiveMatch",
"WatchDownloadedMatch",
"WatchGOTVTheater",
"WatchTwitchStream",
"WeaponInspectExternalItem",
"ViewedOffers",
"BorrowedMusicKit",
"ViewOnMarket",
"InsufficientLevel",
"PartyBrowserRefreshEmpty",
"PartyBrowserRefreshResults",
"PartyBrowserJoin",
"FriendJoin",
"SoloSearch",
"WatchEmbeddedStreamExternally",
"PartyBrowserJoinNearby",
"SwitchedToHRTFEnabled",
"SwitchedToHRTFDisabled",
"PartyBrowserJoinInvited",
};
ConVar steamworks_sessionid_lifetime_client( "steamworks_sessionid_lifetime_client", "0", FCVAR_HIDDEN, "The full client session ID for the new steamworks gamestats." );
CS_App_Lifetime_Gamestats::CS_App_Lifetime_Gamestats() :
BaseClass( "CS_App_Lifetime_Gamestats", "steamworks_sessionid_lifetime_client" ),
m_GameJoinRequested( this, &CS_App_Lifetime_Gamestats::OnGameJoinRequested )
{
}
bool CS_App_Lifetime_Gamestats::Init()
{
if ( !BaseClass::Init() )
return false;
StartSession();
return true;
}
void CS_App_Lifetime_Gamestats::Shutdown()
{
WriteStats();
EndSession();
}
void CS_App_Lifetime_Gamestats::WriteStats( void )
{
// Refresh the interface in case steam has unloaded
m_SteamWorksInterface = GetInterface();
if ( !m_SteamWorksInterface )
{
DevMsg( "WARNING: Attempted to send a steamworks gamestats row when the steamworks interface was not available!" );
}
else
{
WriteUIEvents();
WriteGameJoins();
}
}
EResult CS_App_Lifetime_Gamestats::WriteUIEvents()
{
FOR_EACH_VEC( m_vecUIEvents, i )
{
uint64 iTableID = 0;
m_SteamWorksInterface->AddNewRow( &iTableID, m_SessionID, "CSGOUIEvent" );
if ( !iTableID )
return k_EResultFail;
UIEvent_t *pData = &m_vecUIEvents[i];
WriteStringToTable( pData->strEventID.Access(), iTableID, "EventID" );
WriteIntToTable( pData->unCount, iTableID, "EventCount" );
WriteIntToTable( pData->unTime, iTableID, "TimeSubmitted" );
if ( cl_debug_client_gamestats.GetBool() )
{
Msg( "Steamworks gamestats: %s adding UI data %s id=%d time=%d\n", Name(), pData->strEventID.Get(), pData->unCount, pData->unTime );
}
EResult res = m_SteamWorksInterface->CommitRow( iTableID );
if ( res != k_EResultOK )
{
char pzMessage[MAX_PATH] = {0};
V_snprintf( pzMessage, ARRAYSIZE(pzMessage), "Failed To Submit table %s", "CSUIEvent" );
Assert( pzMessage );
return res;
}
}
m_vecUIEvents.RemoveAll();
return k_EResultOK;
}
EResult CS_App_Lifetime_Gamestats::WriteGameJoins()
{
FOR_EACH_VEC( m_vecGameJoins, i )
{
uint64 iTableID = 0;
m_SteamWorksInterface->AddNewRow( &iTableID, m_SessionID, "CSGOGameJoin" );
if ( !iTableID )
return k_EResultFail;
uint64 ullGameSessionID = m_vecGameJoins[i].ullSessionID;
uint16 unGameType = (uint16)m_vecGameJoins[i].unGameType;
uint16 unGameMode = (uint16)m_vecGameJoins[i].unGameMode;
bool bValveOfficial = m_vecGameJoins[i].bIsValveOfficial;
WriteInt64ToTable( ullGameSessionID, iTableID, "GameSessionID" );
WriteIntToTable( unGameType, iTableID, "GameTypeID" );
WriteIntToTable( unGameMode, iTableID, "GameModeID" );
WriteIntToTable( bValveOfficial, iTableID, "IsValveOfficial" );
if ( cl_debug_client_gamestats.GetBool() )
{
Msg( "Steamworks gamestats: %s adding game session %llu", Name(), ullGameSessionID );
}
EResult res = m_SteamWorksInterface->CommitRow( iTableID );
if ( res != k_EResultOK )
{
char pzMessage[MAX_PATH] = {0};
V_snprintf( pzMessage, ARRAYSIZE(pzMessage), "Failed To Submit table %s", "CSGOGameJoin" );
Assert( pzMessage );
return res;
}
}
m_vecGameJoins.RemoveAll();
return k_EResultOK;
}
static bool BHelperCheckSafeUserCmdString( char const *ipconnect )
{
bool bConnectValid = true;
while ( *ipconnect )
{
if ( ( ( *ipconnect >= '0' ) && ( *ipconnect <= '9' ) ) ||
( ( *ipconnect >= 'a' ) && ( *ipconnect <= 'z' ) ) ||
( ( *ipconnect >= 'A' ) && ( *ipconnect <= 'Z' ) ) ||
( *ipconnect == '_' ) || ( *ipconnect == '-' ) || ( *ipconnect == '.' ) ||
( *ipconnect == ':' ) || ( *ipconnect == '?' ) || ( *ipconnect == '%' ) ||
( *ipconnect == '/' ) || ( *ipconnect == '=' ) || ( *ipconnect == ' ' ) ||
( *ipconnect == '[' ) || ( *ipconnect == ']' ) || ( *ipconnect == '@' ) ||
( *ipconnect == '"' ) || ( *ipconnect == '\'' ) || ( *ipconnect == '#' ) ||
( *ipconnect == '(' ) || ( *ipconnect == ')' ) || ( *ipconnect == '!' ) ||
( *ipconnect == '\\' ) || ( *ipconnect == '$' )
)
++ipconnect;
else
{
bConnectValid = false;
break;
}
}
return bConnectValid;
}
void CS_App_Lifetime_Gamestats::OnGameJoinRequested( GameRichPresenceJoinRequested_t *pCallback )
{
/* Removed for partner depot */
}
void CS_App_Lifetime_Gamestats::RecordUIEvent( const char* szEventName )
{
if ( m_unEventCount == (uint16)-1 )
return;
bool bFound = false;
for ( int i = 0; i < ARRAYSIZE( g_szValidUIEvents ); ++i )
{
if ( V_strcmp( g_szValidUIEvents[i], szEventName) == 0 )
{
bFound = true;
break;
}
}
// Only record whitelisted events
if ( !bFound )
{
return;
}
Assert( V_strlen( szEventName ) < UI_EVENT_NAME_MAX );
uint16 idx = m_vecUIEvents.AddToTail();
m_vecUIEvents[idx].unCount = ++m_unEventCount;
CRTime now;
m_vecUIEvents[idx].unTime = now.GetRTime32();
m_vecUIEvents[idx].strEventID.Set( szEventName );
}
void CS_App_Lifetime_Gamestats::AddSessionIDsToTable( int iTableID )
{
// Our client side session.
WriteInt64ToTable( m_SessionID, iTableID, "SessionID" );
}
void CS_App_Lifetime_Gamestats::RecordGameJoin( uint64 ullSessionID )
{
GameJoins_t gameJoinInfo;
gameJoinInfo.ullSessionID = ullSessionID;
gameJoinInfo.unGameMode = g_pGameTypes->GetCurrentGameMode();
gameJoinInfo.unGameType = g_pGameTypes->GetCurrentGameType();
gameJoinInfo.bIsValveOfficial = CSGameRules() && CSGameRules()->IsValveDS();
if ( m_vecGameJoins.Find( gameJoinInfo ) == m_vecGameJoins.InvalidIndex() )
{
m_vecGameJoins.AddToTail( gameJoinInfo );
}
else
{
Assert( 0 ); // Attempting double add-- would cause PK violation.
}
}