csgo-2018-source/game/server/steamworks_gamestats_server.h
2021-07-24 21:11:47 -07:00

253 lines
7.9 KiB
C++

//====== Copyright © 1996-2006, Valve Corporation, All rights reserved. =======//
//
// Purpose: Uploads gamestats via the SteamWorks API. Server version.
//
//=============================================================================//
#ifndef STEAMWORKS_GAMESTATS_SERVER_H
#define STEAMWORKS_GAMESTATS_SERVER_H
#ifdef _WIN32
#pragma once
#endif
#include "steamworks_gamestats.h"
//used to drive most of the game stat event handlers as well as track basic stats under the hood of CBaseGameStats
class CSteamWorksGameStatsServer : public CSteamWorksGameStatsUploader
{
DECLARE_CLASS( CSteamWorksGameStatsServer, CSteamWorksGameStatsUploader )
public:
CSteamWorksGameStatsServer();
void WriteSessionRow();
protected:
virtual EGameStatsAccountType GetGameStatsAccountType();
// called before a row is committed, allows derived classes to add sessionIDs, etc.
virtual void AddSessionIDsToTable( int iTableID );
};
CSteamWorksGameStatsServer& GetSteamWorksGameStatsServer();
// Macros to ease the creation of SendData method for stats structs/classes
#define BEGIN_STAT_TABLE( tableName ) \
static const char* GetStatTableName( void ) { return tableName; } \
void BuildGamestatDataTable( KeyValues* pKV ) \
{ \
pKV->SetName( GetStatTableName() );
#define REGISTER_STAT( varName ) \
AddDataToKV(pKV, #varName, varName);
#define REGISTER_STAT_NAMED( varName, dbName ) \
AddDataToKV(pKV, dbName, varName);
#define REGISTER_STAT_POSITION( varName ) \
AddPositionDataToKV(pKV, #varName, varName);
#define REGISTER_STAT_POSITION_NAMED( varName, dbName ) \
AddPositionDataToKV(pKV, dbName, varName);
#define REGISTER_STAT_ARRAY( varName ) \
AddArrayDataToKV( pKV, #varName, varName, ARRAYSIZE( varName ) );
#define REGISTER_STAT_ARRAY_NAMED( varName, dbName ) \
AddArrayDataToKV( pKV, dbName, varName, ARRAYSIZE( varName ) );
#define REGISTER_STAT_STRING( varName ) \
AddStringDataToKV( pKV, #varName, varName );
#define REGISTER_STAT_STRING_NAMED( varName, dbName ) \
AddStringDataToKV( pKV, dbName, varName );
#define AUTO_STAT_TABLE_KEY() \
pKV->SetInt( "TimeSubmitted", GetUniqueIDForStatTable( *this ) );
#define END_STAT_TABLE() \
pKV->SetUint64( ::BaseStatData::m_bUseGlobalData ? "TimeSubmitted" : "SessionTime", ::BaseStatData::TimeSubmitted ); \
GetSteamWorksGameStatsServer().AddStatsForUpload( pKV ); \
}
//-----------------------------------------------------------------------------
// Purpose: Templatized class for getting unique ID's for stat tables that need
// to be submitted multiple times per-session.
//-----------------------------------------------------------------------------
template < typename T >
class UniqueStatID_t
{
public:
static unsigned GetNext( void )
{
return ++s_nLastID;
}
static void Reset( void )
{
s_nLastID = 0;
}
private:
static unsigned s_nLastID;
};
template < typename T >
unsigned UniqueStatID_t< T >::s_nLastID = 0;
template < typename T >
unsigned GetUniqueIDForStatTable( const T &table )
{
return UniqueStatID_t< T >::GetNext();
}
//=============================================================================
//
// An interface for tracking gamestats.
//
class IGameStatTracker
{
public:
//-----------------------------------------------------------------------------
// Templatized methods to track a per-mission stat.
// The stat is copied, then deleted after it's sent to the SQL server.
//-----------------------------------------------------------------------------
template < typename T >
void SubmitStat( T& stat )
{
// Make a copy of the stat. All of the stat lists require pointers,
// so we need to protect against a stat allocated on the stack
T* pT = new T();
if( !pT )
return;
*pT = stat;
SubmitStat( pT );
}
//-----------------------------------------------------------------------------
// Templatized methods to track a per-mission stat (by pointer)
// The stat is deleted after it's sent to the SQL server
//-----------------------------------------------------------------------------
template < typename T >
void SubmitStat( T* pStat )
{
// Get the static stat table for this type and add the stat to it
GetStatTable<T>()->AddToTail( pStat );
}
//-----------------------------------------------------------------------------
// Add all stats to an existing key value file for submit.
//-----------------------------------------------------------------------------
virtual void SubmitGameStats( KeyValues *pKV ) = 0;
//-----------------------------------------------------------------------------
// Prints the memory usage of all of the stats being tracked
//-----------------------------------------------------------------------------
void PrintGamestatMemoryUsage( void );
protected:
//=============================================================================
//
// Used as a base interface to store a list of all templatized stat containers
//
class IStatContainer
{
public:
virtual void SendData( KeyValues *pKV ) = 0;
virtual void Clear( void ) = 0;
virtual void PrintMemoryUsage( void ) = 0;
};
// Defines a list of stat containers.
typedef CUtlVector< IStatContainer* > StatContainerList_t;
//-----------------------------------------------------------------------------
// Used to get a list of all stats containers being tracked by the deriving class
//-----------------------------------------------------------------------------
virtual StatContainerList_t* GetStatContainerList( void ) = 0;
private:
//=============================================================================
//
// Templatized list of stats submitted
//
template < typename T >
class CGameStatList : public IStatContainer, public CUtlVector< T* >
{
public:
//-----------------------------------------------------------------------------
// Get data ready to send to the SQL server
//-----------------------------------------------------------------------------
virtual void SendData( KeyValues *pKV )
{
//ASSERT( pKV != NULL );
// Duplicate the master KeyValue for each stat instance
for( int i=0; i < this->m_Size; ++i )
{
// Make a copy of the master key value and build the stat table
KeyValues *pKVCopy = this->operator [](i)->m_bUseGlobalData ? pKV->MakeCopy() : new KeyValues( "" );
this->operator [](i)->BuildGamestatDataTable( pKVCopy );
}
// Reset unique ID counter for the stat type
UniqueStatID_t< T >::Reset();
}
//-----------------------------------------------------------------------------
// Clear and delete every stat in this list
//-----------------------------------------------------------------------------
virtual void Clear( void )
{
this->Purge();
}
//-----------------------------------------------------------------------------
// Print out details about this lists memory usage
//-----------------------------------------------------------------------------
virtual void PrintMemoryUsage( void )
{
if( this->m_Size == 0 )
return;
// Compute the memory used as the size of type times the list count
unsigned uMemoryUsed = this->m_Size * ( sizeof( T ) );
Msg( " %d\tbytes used by %s table\n", uMemoryUsed, T::GetStatTableName() );
}
};
//-----------------------------------------------------------------------------
// Templatized method to get a single instance of a stat list per data type.
//-----------------------------------------------------------------------------
template < typename T >
CGameStatList< T >* GetStatTable( void )
{
static CGameStatList< T > *s_vecOfType = 0;
if( s_vecOfType == 0 )
{
s_vecOfType = new CGameStatList< T >();
GetStatContainerList()->AddToTail( s_vecOfType );
}
return s_vecOfType;
}
};
struct BaseStatData
{
BaseStatData( bool bUseGlobalData = true ) : m_bUseGlobalData( bUseGlobalData )
{
TimeSubmitted = GetSteamWorksGameStatsServer().GetTimeSinceEpoch();
}
bool m_bUseGlobalData;
uint64 TimeSubmitted;
};
#endif // STEAMWORKS_GAMESTATS_SERVER_H