460 lines
11 KiB
C++
460 lines
11 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose:
|
|
//
|
|
// $NoKeywords: $
|
|
//
|
|
//===========================================================================//
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <direct.h>
|
|
#include "tier1/strtools.h"
|
|
#include <conio.h>
|
|
#include "tier1/utlbuffer.h"
|
|
#include "tier2/tier2.h"
|
|
#include "filesystem.h"
|
|
#include "imysqlwrapper.h"
|
|
#include "../../game/server/dod/dod_gamestats.h"
|
|
#include <time.h>
|
|
|
|
|
|
static void Pause( void )
|
|
{
|
|
printf( "Hit a key to continue\n" );
|
|
getch();
|
|
}
|
|
|
|
static void Usage()
|
|
{
|
|
fprintf( stderr, "Usage: gamestats_reader <hostname> <database> <username> <password> <table> <directory to parse> ( -verbose )\n" );
|
|
Pause();
|
|
exit( -1 );
|
|
}
|
|
|
|
#define SQL_CMD_BUFSIZE 16000
|
|
|
|
char sqlCmd[SQL_CMD_BUFSIZE];
|
|
bool g_bFirstCmd;
|
|
bool g_bVerbose;
|
|
|
|
IMySQL *mysql;
|
|
|
|
char **g_argv;
|
|
|
|
void StartMYSQLInsert( void )
|
|
{
|
|
g_bFirstCmd = true;
|
|
sqlCmd[0] = '\0';
|
|
|
|
Q_snprintf( sqlCmd, SQL_CMD_BUFSIZE, "INSERT INTO %s SET ", g_argv[5] );
|
|
}
|
|
|
|
void AddField( const char *field, const char *value )
|
|
{
|
|
char buf[128];
|
|
|
|
if ( !g_bFirstCmd )
|
|
{
|
|
Q_strncat( sqlCmd, ", ", SQL_CMD_BUFSIZE, COPY_ALL_CHARACTERS );
|
|
}
|
|
|
|
Q_snprintf( buf, sizeof(buf), "%s=\"%s\"", field, value );
|
|
Q_strncat( sqlCmd, buf, SQL_CMD_BUFSIZE, COPY_ALL_CHARACTERS );
|
|
|
|
g_bFirstCmd = false;
|
|
}
|
|
|
|
void AddField( const char *field, const int value )
|
|
{
|
|
char buf[128];
|
|
|
|
if ( !g_bFirstCmd )
|
|
{
|
|
Q_strncat( sqlCmd, ", ", SQL_CMD_BUFSIZE, COPY_ALL_CHARACTERS );
|
|
}
|
|
|
|
Q_snprintf( buf, sizeof(buf), "%s=%d", field, value );
|
|
|
|
#ifdef _DEBUG
|
|
if ( Q_strlen(buf) + Q_strlen(sqlCmd) > SQL_CMD_BUFSIZE )
|
|
{
|
|
Assert( !"increase buf size\n" );
|
|
}
|
|
#endif
|
|
|
|
Q_strncat( sqlCmd, buf, SQL_CMD_BUFSIZE, COPY_ALL_CHARACTERS );
|
|
|
|
g_bFirstCmd = false;
|
|
}
|
|
|
|
int CompleteMYSQLInsert( void )
|
|
{
|
|
// terminate command and execute it
|
|
Q_strncat( sqlCmd, "\n", SQL_CMD_BUFSIZE, COPY_ALL_CHARACTERS );
|
|
|
|
if ( g_bVerbose )
|
|
{
|
|
printf( "%s", sqlCmd );
|
|
}
|
|
|
|
int retcode = mysql->Execute( sqlCmd );
|
|
|
|
if ( retcode != 0 )
|
|
{
|
|
const char *asdf = mysql->GetLastError();
|
|
printf( "Error: %s\n", asdf );
|
|
}
|
|
|
|
return retcode;
|
|
}
|
|
|
|
static const char *pszTeamNames[] =
|
|
{
|
|
"allies",
|
|
"axis"
|
|
};
|
|
|
|
static const char *pszClassNames[] =
|
|
{
|
|
"rifleman",
|
|
"assault",
|
|
"support",
|
|
"sniper",
|
|
"mg",
|
|
"rocket"
|
|
};
|
|
|
|
int iDistanceStatWeapons[DOD_NUM_DISTANCE_STAT_WEAPONS] =
|
|
{
|
|
WEAPON_COLT,
|
|
WEAPON_P38,
|
|
WEAPON_C96,
|
|
WEAPON_GARAND,
|
|
WEAPON_GARAND_ZOOMED,
|
|
WEAPON_M1CARBINE,
|
|
WEAPON_K98,
|
|
WEAPON_K98_ZOOMED,
|
|
WEAPON_SPRING,
|
|
WEAPON_SPRING_ZOOMED,
|
|
WEAPON_K98_SCOPED,
|
|
WEAPON_K98_SCOPED_ZOOMED,
|
|
WEAPON_THOMPSON,
|
|
WEAPON_MP40,
|
|
WEAPON_MP44,
|
|
WEAPON_MP44_SEMIAUTO,
|
|
WEAPON_BAR,
|
|
WEAPON_BAR_SEMIAUTO,
|
|
WEAPON_30CAL,
|
|
WEAPON_30CAL_UNDEPLOYED,
|
|
WEAPON_MG42,
|
|
WEAPON_MG42_UNDEPLOYED,
|
|
};
|
|
|
|
// Send hit/shots only for the following weapons
|
|
int iNoDistStatWeapons[DOD_NUM_NODIST_STAT_WEAPONS] =
|
|
{
|
|
WEAPON_AMERKNIFE,
|
|
WEAPON_SPADE,
|
|
WEAPON_BAZOOKA,
|
|
WEAPON_PSCHRECK,
|
|
WEAPON_FRAG_US,
|
|
WEAPON_FRAG_GER,
|
|
WEAPON_FRAG_US_LIVE,
|
|
WEAPON_FRAG_GER_LIVE,
|
|
WEAPON_RIFLEGREN_US,
|
|
WEAPON_RIFLEGREN_GER,
|
|
WEAPON_RIFLEGREN_US_LIVE,
|
|
WEAPON_RIFLEGREN_GER_LIVE,
|
|
WEAPON_THOMPSON_PUNCH,
|
|
WEAPON_MP40_PUNCH,
|
|
};
|
|
|
|
const char * s_WeaponAliasInfo[] =
|
|
{
|
|
"none", // WEAPON_NONE = 0,
|
|
|
|
//Melee
|
|
"amerknife", //WEAPON_AMERKNIFE,
|
|
"spade", //WEAPON_SPADE,
|
|
|
|
//Pistols
|
|
"colt", //WEAPON_COLT,
|
|
"p38", //WEAPON_P38,
|
|
"c96", //WEAPON_C96
|
|
|
|
//Rifles
|
|
"garand", //WEAPON_GARAND,
|
|
"m1carbine", //WEAPON_M1CARBINE,
|
|
"k98", //WEAPON_K98,
|
|
|
|
//Sniper Rifles
|
|
"spring", //WEAPON_SPRING,
|
|
"k98_scoped", //WEAPON_K98_SCOPED,
|
|
|
|
//SMG
|
|
"thompson", //WEAPON_THOMPSON,
|
|
"mp40", //WEAPON_MP40,
|
|
"mp44", //WEAPON_MP44,
|
|
"bar", //WEAPON_BAR,
|
|
|
|
//Machine guns
|
|
"30cal", //WEAPON_30CAL,
|
|
"mg42", //WEAPON_MG42,
|
|
|
|
//Rocket weapons
|
|
"bazooka", //WEAPON_BAZOOKA,
|
|
"pschreck", //WEAPON_PSCHRECK,
|
|
|
|
//Grenades
|
|
"frag_us", //WEAPON_FRAG_US,
|
|
"frag_ger", //WEAPON_FRAG_GER,
|
|
|
|
"frag_us_live", //WEAPON_FRAG_US_LIVE
|
|
"frag_ger_live", //WEAPON_FRAG_GER_LIVE
|
|
|
|
"smoke_us", //WEAPON_SMOKE_US
|
|
"smoke_ger", //WEAPON_SMOKE_GER
|
|
|
|
"riflegren_us", //WEAPON_RIFLEGREN_US
|
|
"riflegren_ger", //WEAPON_RIFLEGREN_GER
|
|
|
|
"riflegren_us_live", //WEAPON_RIFLEGREN_US_LIVE
|
|
"riflegren_ger_live", //WEAPON_RIFLEGREN_GER_LIVE
|
|
|
|
// not actually separate weapons, but defines used in stats recording
|
|
"thompson_punch", //WEAPON_THOMPSON_PUNCH
|
|
"mp40_punch", //WEAPON_MP40_PUNCH
|
|
"garand_zoomed", //WEAPON_GARAND_ZOOMED,
|
|
|
|
"k98_zoomed", //WEAPON_K98_ZOOMED
|
|
"spring_zoomed", //WEAPON_SPRING_ZOOMED
|
|
"k98_scoped_zoomed", //WEAPON_K98_SCOPED_ZOOMED
|
|
|
|
"30cal_undeployed", //WEAPON_30CAL_UNDEPLOYED,
|
|
"mg42_undeployed", //WEAPON_MG42_UNDEPLOYED,
|
|
|
|
"bar_semiauto", //WEAPON_BAR_SEMIAUTO,
|
|
"mp44_semiauto", //WEAPON_MP44_SEMIAUTO,
|
|
|
|
0, // end of list marker
|
|
};
|
|
|
|
void ParseFile( const char *fileName )
|
|
{
|
|
FileHandle_t file = g_pFullFileSystem->Open( fileName, "rb" );
|
|
|
|
if ( !file )
|
|
{
|
|
return;
|
|
}
|
|
|
|
dod_gamestats_t stats;
|
|
g_pFullFileSystem->Read( &stats, sizeof( dod_gamestats_t ), file );
|
|
|
|
if ( stats.header.iVersion != DOD_STATS_BLOB_VERSION || Q_stricmp( stats.header.szGameName, "dod" ) )
|
|
{
|
|
printf( "Error parsing file, bad header info: %s\n", fileName );
|
|
return;
|
|
}
|
|
|
|
StartMYSQLInsert();
|
|
|
|
AddField( "map", stats.header.szMapName );
|
|
|
|
const time_t mapfiletime = g_pFullFileSystem->GetFileTime( fileName );
|
|
|
|
struct tm *t = localtime( &mapfiletime );
|
|
|
|
// YYYY-MM-DD HH:MM::SS
|
|
|
|
char filetimebuf[64];
|
|
|
|
Q_snprintf( filetimebuf, sizeof(filetimebuf), "%04d-%02d-%02d %02d:%02d:%02d",
|
|
t->tm_year + 1900,
|
|
t->tm_mon + 1,
|
|
t->tm_mday,
|
|
t->tm_hour,
|
|
t->tm_min,
|
|
t->tm_sec );
|
|
|
|
AddField( "time", filetimebuf );
|
|
|
|
AddField( "version", stats.header.iVersion );
|
|
|
|
AddField( "ipaddr_0", stats.header.ipAddr[0] );
|
|
AddField( "ipaddr_1", stats.header.ipAddr[1] );
|
|
AddField( "ipaddr_2", stats.header.ipAddr[2] );
|
|
AddField( "ipaddr_3", stats.header.ipAddr[3] );
|
|
AddField( "port", stats.header.port );
|
|
|
|
AddField( "minutes_map", stats.iMinutesPlayed );
|
|
AddField( "wins_allies", stats.iNumAlliesWins );
|
|
AddField( "wins_axis", stats.iNumAxisWins);
|
|
AddField( "tickpoints_allies", stats.iAlliesTickPoints );
|
|
AddField( "tickpoints_axis", stats.iAxisTickPoints );
|
|
|
|
char buf[128];
|
|
|
|
for ( int cls=0;cls<6;cls++ )
|
|
{
|
|
Q_snprintf( buf, sizeof(buf), "minutes_allies_%s", pszClassNames[cls] );
|
|
AddField( buf, stats.iMinutesPlayedPerClass_Allies[cls] );
|
|
|
|
Q_snprintf( buf, sizeof(buf), "kills_allies_%s", pszClassNames[cls] );
|
|
AddField( buf, stats.iKillsPerClass_Allies[cls] );
|
|
|
|
Q_snprintf( buf, sizeof(buf), "defenses_allies_%s", pszClassNames[cls] );
|
|
AddField( buf, stats.iDefensesPerClass_Allies[cls] );
|
|
|
|
Q_snprintf( buf, sizeof(buf), "caps_allies_%s", pszClassNames[cls] );
|
|
AddField( buf, stats.iCapsPerClass_Allies[cls] );
|
|
|
|
Q_snprintf( buf, sizeof(buf), "spawns_allies_%s", pszClassNames[cls] );
|
|
AddField( buf, stats.iSpawnsPerClass_Allies[cls] );
|
|
|
|
Q_snprintf( buf, sizeof(buf), "classlimit_allies_%s", pszClassNames[cls] );
|
|
AddField( buf, stats.iClassLimits_Allies[cls] );
|
|
|
|
|
|
Q_snprintf( buf, sizeof(buf), "minutes_axis_%s", pszClassNames[cls] );
|
|
AddField( buf, stats.iMinutesPlayedPerClass_Axis[cls] );
|
|
|
|
Q_snprintf( buf, sizeof(buf), "kills_axis_%s", pszClassNames[cls] );
|
|
AddField( buf, stats.iKillsPerClass_Axis[cls] );
|
|
|
|
Q_snprintf( buf, sizeof(buf), "defenses_axis_%s", pszClassNames[cls] );
|
|
AddField( buf, stats.iDefensesPerClass_Axis[cls] );
|
|
|
|
Q_snprintf( buf, sizeof(buf), "caps_axis_%s", pszClassNames[cls] );
|
|
AddField( buf, stats.iCapsPerClass_Axis[cls] );
|
|
|
|
Q_snprintf( buf, sizeof(buf), "spawns_axis_%s", pszClassNames[cls] );
|
|
AddField( buf, stats.iSpawnsPerClass_Axis[cls] );
|
|
|
|
Q_snprintf( buf, sizeof(buf), "classlimit_axis_%s", pszClassNames[cls] );
|
|
AddField( buf, stats.iClassLimits_Axis[cls] );
|
|
}
|
|
|
|
int i;
|
|
|
|
for ( i=0;i<DOD_NUM_DISTANCE_STAT_WEAPONS;i++ )
|
|
{
|
|
int iWeapon = iDistanceStatWeapons[i];
|
|
const char *pszWeapon = s_WeaponAliasInfo[iWeapon];
|
|
|
|
Q_snprintf( buf, sizeof(buf), "weapon_shots_%s", pszWeapon );
|
|
AddField( buf, stats.weaponStatsDistance[i].iNumAttacks );
|
|
|
|
Q_snprintf( buf, sizeof(buf), "weapon_hits_%s", pszWeapon );
|
|
AddField( buf, stats.weaponStatsDistance[i].iNumHits );
|
|
|
|
for ( int j=0;j<DOD_NUM_WEAPON_DISTANCE_BUCKETS;j++ )
|
|
{
|
|
Q_snprintf( buf, sizeof(buf), "weapon_distance_%s_%d", pszWeapon, j );
|
|
AddField( buf, stats.weaponStatsDistance[i].iDistanceBuckets[j] );
|
|
}
|
|
}
|
|
|
|
for ( i=0;i<DOD_NUM_NODIST_STAT_WEAPONS;i++ )
|
|
{
|
|
int iWeapon = iNoDistStatWeapons[i];
|
|
const char *pszWeapon = s_WeaponAliasInfo[iWeapon];
|
|
|
|
Q_snprintf( buf, sizeof(buf), "weapon_shots_%s", pszWeapon );
|
|
AddField( buf, stats.weaponStats[i].iNumAttacks );
|
|
|
|
Q_snprintf( buf, sizeof(buf), "weapon_hits_%s", pszWeapon );
|
|
AddField( buf, stats.weaponStats[i].iNumHits );
|
|
}
|
|
|
|
CompleteMYSQLInsert();
|
|
|
|
g_pFullFileSystem->Close( file );
|
|
}
|
|
|
|
int main( int argc, char **argv )
|
|
{
|
|
g_argv = argv;
|
|
|
|
if( argc < 6 )
|
|
{
|
|
Usage();
|
|
}
|
|
|
|
if ( argc == 7 && !Q_stricmp( argv[6], "-verbose" ) )
|
|
{
|
|
g_bVerbose = true;
|
|
}
|
|
|
|
InitDefaultFileSystem();
|
|
|
|
// Init MYSQL connection
|
|
CSysModule *sql = Sys_LoadModule( "mysql_wrapper" );
|
|
if ( sql )
|
|
{
|
|
CreateInterfaceFn factory = Sys_GetFactory( sql );
|
|
if ( factory )
|
|
{
|
|
mysql = ( IMySQL * )factory( MYSQL_WRAPPER_VERSION_NAME, NULL );
|
|
if ( mysql )
|
|
{
|
|
if ( mysql->InitMySQL( argv[ 2 ], argv[ 1 ], argv[ 3 ], argv[ 4 ] ) )
|
|
{
|
|
// insert rows
|
|
|
|
const char *dir = argv[6];
|
|
|
|
char searchString[MAX_PATH*2];
|
|
Q_strncpy( searchString, dir, sizeof( searchString ) );
|
|
Q_AppendSlash( searchString, sizeof( searchString ) );
|
|
Q_strncat( searchString, "*.dat", sizeof( searchString ), COPY_ALL_CHARACTERS );
|
|
|
|
int iNumFiles = 0;
|
|
FileFindHandle_t findHandle = NULL;
|
|
const char *filename = g_pFullFileSystem->FindFirst( searchString, &findHandle );
|
|
while ( filename )
|
|
{
|
|
char fullFileName[MAX_PATH*2];
|
|
|
|
Q_strncpy( fullFileName, dir, sizeof( fullFileName ) );
|
|
Q_AppendSlash( fullFileName, sizeof( fullFileName ) );
|
|
Q_strncat( fullFileName, filename, sizeof( fullFileName ), COPY_ALL_CHARACTERS );
|
|
|
|
ParseFile( fullFileName );
|
|
|
|
printf( "processing file: %s\n", fullFileName );
|
|
iNumFiles++;
|
|
|
|
filename = g_pFullFileSystem->FindNext(findHandle);
|
|
}
|
|
g_pFullFileSystem->FindClose(findHandle);
|
|
|
|
printf( "Completed: %d files processed from directory \"%s\"\n", iNumFiles, dir );
|
|
|
|
}
|
|
else
|
|
{
|
|
printf( "InitMySQL failed ( %s )\n", mysql->GetLastError() );
|
|
}
|
|
|
|
mysql->Release();
|
|
}
|
|
else
|
|
{
|
|
printf( "Unable to connect via mysql_wrapper\n");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
printf( "Unable to get factory from mysql_wrapper.dll, not updating access mysql table!!!" );
|
|
}
|
|
|
|
Sys_UnloadModule( sql );
|
|
}
|
|
else
|
|
{
|
|
printf( "Unable to load mysql_wrapper.dll, not updating access mysql table!!!" );
|
|
}
|
|
|
|
return 0;
|
|
}
|