460 lines
13 KiB
C++
460 lines
13 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose: Implemenatation of CMatchInfo
|
|
//
|
|
// $Workfile: $
|
|
// $Date: $
|
|
//
|
|
//------------------------------------------------------------------------------------------------------
|
|
// $Log: $
|
|
//
|
|
// $NoKeywords: $
|
|
//=============================================================================//
|
|
#include "MatchInfo.h"
|
|
|
|
CMatchInfo* g_pMatchInfo=NULL; //global information about the match.
|
|
|
|
//------------------------------------------------------------------------------------------------------
|
|
// Function: CMatchInfo::generate
|
|
// Purpose: generates the match info structure from the log file
|
|
//------------------------------------------------------------------------------------------------------
|
|
void CMatchInfo::generate()
|
|
{
|
|
if (plogfile->empty())
|
|
g_pApp->fatalError("No data in log file!\nPlease ensure that you are running TFstats on a valid log file!");
|
|
|
|
CEventListIterator it=plogfile->begin();
|
|
logopentime=(*it)->getTime();
|
|
for (it;it!=plogfile->end();++it)
|
|
{
|
|
const CLogEvent* curr=(*it);
|
|
switch(curr->getType())
|
|
{
|
|
case CLogEvent::CONNECT:
|
|
{
|
|
int sid=curr->getArgument(0)->asPlayerGetSvrPID();
|
|
unsigned long WONid=curr->getArgument(0)->asPlayerGetWONID();
|
|
string plrName=curr->getArgument(0)->asPlayerGetName();
|
|
string ipAddress=curr->getArgument(1)->getStringValue();
|
|
PID pid;
|
|
PID foundpid=-1;
|
|
if (WONid!=-1)
|
|
{
|
|
bLanGame=false;
|
|
pid=pidMap[sid]=WONid;
|
|
}
|
|
else
|
|
{
|
|
bLanGame=true;
|
|
|
|
CPlayerList::iterator it=players.begin();
|
|
for (it;it!=players.end();++it)
|
|
{
|
|
PID currpid=it->first;
|
|
CPlayer& cp=it->second;
|
|
if (cp.ipAddress==ipAddress)
|
|
{
|
|
foundpid=currpid;
|
|
break;
|
|
}
|
|
}
|
|
if (it==players.end()) //if no ip addresses matched match by name
|
|
{
|
|
it = players.begin();
|
|
for (it;it!=players.end();++it)
|
|
{
|
|
PID currpid=it->first;
|
|
CPlayer& cp=it->second;
|
|
if (cp.aliases.contains(plrName))
|
|
{
|
|
foundpid=currpid;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (foundpid != -1)
|
|
{
|
|
pid=pidMap[sid]=foundpid;
|
|
}
|
|
else
|
|
{
|
|
pid=pidMap[sid]=sid;
|
|
}
|
|
//printf("Checkpoint %lu\n",__LINE__);
|
|
//printf("pid=%lu\n",pid);
|
|
if (players[pid].pid==-1)
|
|
players[pid].pid=pid;
|
|
|
|
players[pid].ipAddress=ipAddress;
|
|
players[pid].svrPID=sid;
|
|
players[pid].WONID=WONid;
|
|
//keep the pseudonym list updated
|
|
players[pid].nameFound(curr->getTime(),plrName);
|
|
|
|
}
|
|
break;
|
|
case CLogEvent::ENTER_GAME:
|
|
{
|
|
int sid=curr->getArgument(0)->asPlayerGetSvrPID();
|
|
//PID pid=curr->getArgument(0)->asPlayerGetFullPID();
|
|
unsigned long WONid=curr->getArgument(0)->asPlayerGetWONID();
|
|
|
|
PID pid;
|
|
if (WONid!=-1)
|
|
{
|
|
bLanGame=false;
|
|
pid=pidMap[sid]=WONid;
|
|
}
|
|
else
|
|
{
|
|
bLanGame=true;
|
|
|
|
//they may have matched based on IP or name.
|
|
//so check if the player structure pointed to by
|
|
//the sid is valid, if so, don't reassign pid
|
|
pid=pidMap[sid];
|
|
if (players[pid].ipAddress=="")
|
|
pid=pidMap[sid]=sid;
|
|
}
|
|
|
|
players[pid].svrPID=sid;
|
|
players[pid].WONID=WONid;
|
|
players[pid].pid=pid;
|
|
string nm=curr->getArgument(0)->asPlayerGetName();
|
|
|
|
//keep the pseudonym list updated
|
|
players[pid].nameFound(curr->getTime(),nm);
|
|
}
|
|
break;
|
|
case CLogEvent::CLASS_CHANGE:
|
|
{
|
|
PID pid=curr->getArgument(0)->asPlayerGetPID();
|
|
time_t changetime=curr->getTime();
|
|
|
|
player_class newpc=playerClassNameToClassID(curr->getArgument(1)->getStringValue());
|
|
|
|
//keep the pseudonym list updated
|
|
players[pid].nameFound(curr->getTime(),curr->getArgument(0)->asPlayerGetName());
|
|
string plrname=curr->getArgument(0)->asPlayerGetName();
|
|
|
|
players[pid].allclassesplayed.add(changetime,newpc);
|
|
|
|
int currTeam=players[pid].teams.atTime(changetime);
|
|
players[pid].perteam[currTeam].classesplayed.add(changetime,newpc);
|
|
|
|
}
|
|
break;
|
|
case CLogEvent::NAME_CHANGE:
|
|
{
|
|
//keep the pseudonym list updated
|
|
players[curr->getArgument(0)->asPlayerGetPID()].nameFound(curr->getTime(),curr->getArgument(1)->asPlayerGetName());
|
|
}
|
|
break;
|
|
case CLogEvent::SUICIDE:
|
|
{
|
|
PID pid=(*it)->getArgument(0)->asPlayerGetPID();
|
|
int team=players[pid].teams.atTime((*it)->getTime());
|
|
|
|
// players[pid].perteam[team].kills++;
|
|
players[pid].perteam[team].deaths++;
|
|
players[pid].perteam[team].suicides++;
|
|
|
|
//keep the pseudonym list updated
|
|
players[pid].nameFound(curr->getTime(),curr->getArgument(0)->asPlayerGetName());
|
|
}
|
|
break;
|
|
case CLogEvent::FRAG:
|
|
case CLogEvent::TEAM_FRAG:
|
|
{
|
|
PID killerid=(*it)->getArgument(0)->asPlayerGetPID();
|
|
PID killedid=(*it)->getArgument(1)->asPlayerGetPID();
|
|
int killerTeam=players[killerid].teams.atTime((*it)->getTime());
|
|
int killedTeam=players[killedid].teams.atTime((*it)->getTime());
|
|
|
|
|
|
CPlayer& p1=players[killerid];
|
|
CPlayer& p2=players[killedid];
|
|
|
|
|
|
if (curr->getType() == CLogEvent::TEAM_FRAG)
|
|
{
|
|
players[killerid].perteam[killerTeam].teamkills++;
|
|
players[killedid].perteam[killedTeam].teamkilled++;
|
|
}
|
|
else if (curr->getType() == CLogEvent::FRAG)
|
|
{
|
|
string weapName=(*it)->getArgument(2)->getStringValue();
|
|
|
|
bool countKill=true;
|
|
|
|
//gotta account for timer/infection double kills for medics!
|
|
if (weapName=="infection")
|
|
{
|
|
//test to see if the previous event was a timer from the same player, and a kill, and with the timer.
|
|
CEventListIterator it2=it;
|
|
if ((--it2)!=plogfile->begin())
|
|
{
|
|
if ((*it2)->getType() == CLogEvent::FRAG)
|
|
if ((*it2)->getArgument(2)->getStringValue()=="timer")
|
|
if ((*it2)->getArgument(0)->asPlayerGetPID()==killerid)
|
|
countKill=false;
|
|
}
|
|
}
|
|
if (countKill)
|
|
{
|
|
|
|
players[killerid].perteam[killerTeam].weaponKills[weapName]++;
|
|
players[killerid].perteam[killerTeam].kills++;
|
|
players[killedid].perteam[killedTeam].deaths++;
|
|
}
|
|
}
|
|
|
|
//keep the pseudonym list updated
|
|
players[killerid].nameFound(curr->getTime(),curr->getArgument(0)->asPlayerGetName());
|
|
players[killedid].nameFound(curr->getTime(),curr->getArgument(1)->asPlayerGetName());
|
|
|
|
}
|
|
break;
|
|
case CLogEvent::TEAM_JOIN:
|
|
{
|
|
int team=curr->getArgument(1)->getFloatValue();
|
|
team--; //teams are logged as 1-4. tfstats stores them as 0-3
|
|
PID pid=curr->getArgument(0)->asPlayerGetPID();
|
|
|
|
|
|
CPlayer& p=players[pid];
|
|
team_exists[team]=true;
|
|
|
|
int oldteam=team;
|
|
if(p.teams.anythingAtTime(curr->getTime()-1))
|
|
oldteam=p.teams.atTime(curr->getTime()-1);
|
|
else //if this is the first team join, count them as in the game
|
|
players[pid].logontime=curr->getTime();
|
|
|
|
//keep the pseudonym list updated
|
|
players[pid].nameFound(curr->getTime(),curr->getArgument(0)->asPlayerGetName());
|
|
|
|
players[pid].teams.add(curr->getTime(),team);
|
|
|
|
if (p.allclassesplayed.anythingAtTime(curr->getTime()))
|
|
{
|
|
player_class plrcurrclass=players[pid].allclassesplayed.atTime(curr->getTime());
|
|
players[pid].perteam[oldteam].classesplayed.cut(curr->getTime());
|
|
players[pid].perteam[team].classesplayed.add(curr->getTime(),plrcurrclass);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case CLogEvent::TEAM_RENAME:
|
|
{
|
|
int teamid=curr->getArgument(0)->getFloatValue()-1;
|
|
string tname=curr->getArgument(1)->getStringValue();
|
|
teamnames[teamid]=tname;
|
|
}
|
|
break;
|
|
|
|
case CLogEvent::SERVER_NAME:
|
|
{
|
|
servername=curr->getArgument(0)->getStringValue();
|
|
}
|
|
break;
|
|
case CLogEvent::SERVER_SPAWN:
|
|
{
|
|
mapname=curr->getArgument(0)->getStringValue();
|
|
}
|
|
break;
|
|
case CLogEvent::DISCONNECT:
|
|
{
|
|
PID pid=curr->getArgument(0)->asPlayerGetPID();
|
|
players[pid].logofftime=curr->getTime();
|
|
players[pid].allclassesplayed.endTime=curr->getTime();
|
|
players[pid].allclassesplayed.cut(curr->getTime());
|
|
players[pid].teams.cut(curr->getTime());
|
|
players[pid].aliases.cut(curr->getTime());
|
|
|
|
int currTeam=players[pid].teams.atTime(curr->getTime());
|
|
players[pid].perteam[currTeam].classesplayed.cut(curr->getTime());
|
|
|
|
//keep the pseudonym list updated
|
|
if (pid!=-1) //sometimes disconnect messages have -1 for the pid
|
|
players[pid].nameFound(curr->getTime(),curr->getArgument(0)->asPlayerGetName());
|
|
|
|
}
|
|
break;
|
|
case CLogEvent::NAMED_BROADCAST:
|
|
{
|
|
//keep the pseudonym list updated
|
|
const CLogEventArgument* pArg=curr->getArgument(1);
|
|
PID pid=pArg->asPlayerGetPID();
|
|
players[pid].nameFound(curr->getTime(),curr->getArgument(1)->asPlayerGetName());
|
|
}
|
|
break;
|
|
case CLogEvent::NAMED_GOAL_ACTIVATE:
|
|
{
|
|
//keep the pseudonym list updated
|
|
players[curr->getArgument(0)->asPlayerGetPID()].nameFound(curr->getTime(),curr->getArgument(0)->asPlayerGetName());
|
|
}
|
|
break;
|
|
case CLogEvent::LOG_CLOSED:
|
|
{
|
|
logclosetime=curr->getTime();
|
|
}
|
|
break;
|
|
}
|
|
|
|
#ifdef _DEBUG
|
|
#ifdef _PARSEDEBUG
|
|
|
|
printf("%s:\n",CLogEvent::TypeNames[(int)curr->getType()]);
|
|
fflush(stdout);
|
|
printf("\t%s\n",curr->m_StrippedText);
|
|
fflush(stdout);
|
|
for (int i=0;curr->getArgument(i);i++)
|
|
{
|
|
if (i==0)
|
|
printf("\t\targs: ");
|
|
fflush(stdout);
|
|
printf("\"%s\" ",curr->getArgument(i)->getStringValue());
|
|
}
|
|
printf("\n");
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
|
|
if (logclosetime==0 && !plogfile->empty())
|
|
{
|
|
CEventListIterator it=plogfile->end();
|
|
--it;
|
|
logclosetime=(*it)->getTime();
|
|
}
|
|
|
|
map<PID,CPlayer>::iterator it2;
|
|
for(it2=players.begin();it2!=players.end();++it2)
|
|
{
|
|
CPlayer& p=(*it2).second;
|
|
|
|
|
|
if (p.aliases.endTime < logclosetime)
|
|
p.aliases.endTime=logclosetime;
|
|
|
|
p.name=p.aliases.favourite();
|
|
|
|
if (p.allclassesplayed.endTime < logclosetime)
|
|
p.allclassesplayed.endTime=logclosetime;
|
|
|
|
if (p.teams.endTime < logclosetime)
|
|
p.teams.endTime=logclosetime;
|
|
|
|
for (int i=0;i<MAX_TEAMS;i++)
|
|
{
|
|
//if you have no kills you have to play on a team at least 30 seconds to be counted part of it
|
|
//also give a one-suicide grace so they can killthemselves to get onto another team?
|
|
if (p.teams.howLong(i) < 30 && p.perteam[i].kills==0)// && p.perteam[i].deaths < 1)
|
|
p.teams.remove(i);
|
|
|
|
CTimeIndexedList<player_class>* v= &p.perteam[i].classesplayed;
|
|
p.perteam[i].classesplayed.endTime=logclosetime;
|
|
|
|
time_t t=p.teams.howLong(i);
|
|
p.perteam[i].timeon=t;
|
|
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
|
|
//------------------------------------------------------------------------------------------------------
|
|
// Function: CMatchInfo::getPlayerID
|
|
// Purpose: resolves a player name to that players PID
|
|
// Input: name - the name
|
|
// Output: PID the PID
|
|
//------------------------------------------------------------------------------------------------------
|
|
PID CMatchInfo::getPlayerID(string name)
|
|
{
|
|
CPlayerListIterator it;
|
|
|
|
//ugh! O(n)
|
|
for (it=playerBegin();it!=playerEnd();++it)
|
|
{
|
|
PID id=(*it).first;
|
|
CPlayer curr=(*it).second;
|
|
|
|
if (curr.name == name)
|
|
return id;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/*
|
|
unsigned long CMatchInfo::getPlayerWONID(string name)
|
|
{
|
|
CPlayerListIterator it;
|
|
|
|
//ugh! O(n)
|
|
for (it=playerBegin();it!=playerEnd();++it)
|
|
{
|
|
CPlayer curr=(*it).second;
|
|
|
|
if (curr.name == name)
|
|
return curr.WONID;
|
|
}
|
|
return 0xffffffff;
|
|
}
|
|
*/
|
|
//------------------------------------------------------------------------------------------------------
|
|
// Function: CMatchInfo::CMatchInfo
|
|
// Purpose: Constructor
|
|
// Input: plf - the log file
|
|
// Output:
|
|
//------------------------------------------------------------------------------------------------------
|
|
CMatchInfo::CMatchInfo(CEventList* plf)
|
|
:numPlrs(0),logclosetime(0),plogfile(plf)
|
|
{
|
|
teamnames[0]="Blue";
|
|
teamnames[1]="Red";
|
|
teamnames[2]="Yellow";
|
|
teamnames[3]="Green";
|
|
team_exists[0]=team_exists[1]=team_exists[2]=team_exists[3]=false;
|
|
|
|
|
|
generate();
|
|
|
|
}
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------------------------------------
|
|
// Function: CMatchInfo::teamID
|
|
// Purpose: resolves a team name to its ID
|
|
// Input: teamname - the team name
|
|
// Output: int the ID of the team
|
|
//------------------------------------------------------------------------------------------------------
|
|
int CMatchInfo::teamID(string teamname)
|
|
{
|
|
for (int i=0;i<MAX_TEAMS;i++)
|
|
if (stricmp(teamname.c_str(),teamnames[i].c_str())==0)
|
|
return i;
|
|
return -1;
|
|
}
|
|
|
|
|
|
//------------------------------------------------------------------------------------------------------
|
|
// Function: CMatchInfo::getTimeOn
|
|
// Purpose: returns how long the specified player has been playing
|
|
// Input: pid - the player being queried
|
|
// Output: time_t the time he/she played
|
|
//------------------------------------------------------------------------------------------------------
|
|
time_t CMatchInfo::getTimeOn(PID pid)
|
|
{
|
|
CPlayer& p=players[pid];
|
|
|
|
if (p.logofftime==0)
|
|
p.logofftime=logclosetime;
|
|
|
|
time_t timeon=p.logofftime-p.logontime;
|
|
|
|
return timeon;
|
|
}
|