mirror of
https://github.com/ddnet/ddnet.git
synced 2024-11-14 12:08:20 +00:00
524 lines
12 KiB
C++
524 lines
12 KiB
C++
#include "teehistorian.h"
|
|
|
|
#include <engine/shared/config.h>
|
|
#include <engine/shared/snapshot.h>
|
|
#include <game/gamecore.h>
|
|
|
|
static const char TEEHISTORIAN_NAME[] = "teehistorian@ddnet.tw";
|
|
static const CUuid TEEHISTORIAN_UUID = CalculateUuid(TEEHISTORIAN_NAME);
|
|
static const char TEEHISTORIAN_VERSION[] = "1";
|
|
|
|
enum
|
|
{
|
|
TEEHISTORIAN_NONE,
|
|
TEEHISTORIAN_FINISH,
|
|
TEEHISTORIAN_TICK_SKIP,
|
|
TEEHISTORIAN_PLAYER_NEW,
|
|
TEEHISTORIAN_PLAYER_OLD,
|
|
TEEHISTORIAN_INPUT_DIFF,
|
|
TEEHISTORIAN_INPUT_NEW,
|
|
TEEHISTORIAN_MESSAGE,
|
|
TEEHISTORIAN_JOIN,
|
|
TEEHISTORIAN_DROP,
|
|
TEEHISTORIAN_CONSOLE_COMMAND,
|
|
};
|
|
|
|
static char EscapeJsonChar(char c)
|
|
{
|
|
switch(c)
|
|
{
|
|
case '\"': return '\"';
|
|
case '\\': return '\\';
|
|
case '\b': return 'b';
|
|
case '\n': return 'n';
|
|
case '\r': return 'r';
|
|
case '\t': return 't';
|
|
// Don't escape '\f', who uses that. :)
|
|
default: return 0;
|
|
}
|
|
}
|
|
|
|
static char *EscapeJson(char *pBuffer, int BufferSize, const char *pString)
|
|
{
|
|
dbg_assert(BufferSize > 0, "can't null-terminate the string");
|
|
// Subtract the space for null termination early.
|
|
BufferSize--;
|
|
|
|
char *pResult = pBuffer;
|
|
while(BufferSize && *pString)
|
|
{
|
|
char c = *pString;
|
|
pString++;
|
|
char Escaped = EscapeJsonChar(c);
|
|
if(Escaped)
|
|
{
|
|
if(BufferSize < 2)
|
|
{
|
|
break;
|
|
}
|
|
*pBuffer++ = '\\';
|
|
*pBuffer++ = Escaped;
|
|
BufferSize -= 2;
|
|
}
|
|
// Assuming ASCII/UTF-8, "if control character".
|
|
else if(c < 0x20)
|
|
{
|
|
// \uXXXX
|
|
if(BufferSize < 6)
|
|
{
|
|
break;
|
|
}
|
|
str_format(pBuffer, BufferSize, "\\u%04x", c);
|
|
BufferSize -= 6;
|
|
}
|
|
else
|
|
{
|
|
*pBuffer++ = c;
|
|
}
|
|
}
|
|
*pBuffer = 0;
|
|
return pResult;
|
|
}
|
|
|
|
CTeeHistorian::CTeeHistorian()
|
|
{
|
|
m_State = STATE_START;
|
|
m_pfnWriteCallback = 0;
|
|
m_pWriteCallbackUserdata = 0;
|
|
}
|
|
|
|
void CTeeHistorian::Reset(const CGameInfo *pGameInfo, WRITE_CALLBACK pfnWriteCallback, void *pUser)
|
|
{
|
|
dbg_assert(m_State == STATE_START || m_State == STATE_BEFORE_TICK, "invalid teehistorian state");
|
|
|
|
m_Debug = 0;
|
|
|
|
m_Tick = 0;
|
|
m_LastWrittenTick = 0;
|
|
// `m_TickWritten` is initialized in `BeginTick`
|
|
m_MaxClientID = MAX_CLIENTS;
|
|
// `m_PrevMaxClientID` is initialized in `BeginTick`
|
|
for(int i = 0; i < MAX_CLIENTS; i++)
|
|
{
|
|
m_aPrevPlayers[i].m_Alive = false;
|
|
m_aPrevPlayers[i].m_InputExists = false;
|
|
}
|
|
m_pfnWriteCallback = pfnWriteCallback;
|
|
m_pWriteCallbackUserdata = pUser;
|
|
|
|
WriteHeader(pGameInfo);
|
|
|
|
m_State = STATE_START;
|
|
}
|
|
|
|
void CTeeHistorian::WriteHeader(const CGameInfo *pGameInfo)
|
|
{
|
|
Write(&TEEHISTORIAN_UUID, sizeof(TEEHISTORIAN_UUID));
|
|
|
|
char aGameUuid[UUID_MAXSTRSIZE];
|
|
char aStartTime[128];
|
|
|
|
FormatUuid(pGameInfo->m_GameUuid, aGameUuid, sizeof(aGameUuid));
|
|
str_timestamp_ex(pGameInfo->m_StartTime, aStartTime, sizeof(aStartTime), "%Y-%m-%d %H:%M:%S %z");
|
|
|
|
char aCommentBuffer[128];
|
|
char aServerVersionBuffer[128];
|
|
char aStartTimeBuffer[128];
|
|
char aServerNameBuffer[128];
|
|
char aGameTypeBuffer[128];
|
|
char aMapNameBuffer[128];
|
|
|
|
char aJson[2048];
|
|
|
|
#define E(buf, str) EscapeJson(buf, sizeof(buf), str)
|
|
|
|
str_format(aJson, sizeof(aJson), "{\"comment\":\"%s\",\"version\":\"%s\",\"game_uuid\":\"%s\",\"server_version\":\"%s\",\"start_time\":\"%s\",\"server_name\":\"%s\",\"server_port\":\"%d\",\"game_type\":\"%s\",\"map_name\":\"%s\",\"map_size\":\"%d\",\"map_crc\":\"%08x\",\"config\":{",
|
|
E(aCommentBuffer, TEEHISTORIAN_NAME),
|
|
TEEHISTORIAN_VERSION,
|
|
aGameUuid,
|
|
E(aServerVersionBuffer, pGameInfo->m_pServerVersion),
|
|
E(aStartTimeBuffer, aStartTime),
|
|
E(aServerNameBuffer, pGameInfo->m_pServerName),
|
|
pGameInfo->m_ServerPort,
|
|
E(aGameTypeBuffer, pGameInfo->m_pGameType),
|
|
E(aMapNameBuffer, pGameInfo->m_pMapName),
|
|
pGameInfo->m_MapSize,
|
|
pGameInfo->m_MapCrc);
|
|
Write(aJson, str_length(aJson));
|
|
|
|
char aBuffer1[1024];
|
|
char aBuffer2[1024];
|
|
bool First = true;
|
|
|
|
#define MACRO_CONFIG_INT(Name,ScriptName,Def,Min,Max,Flags,Desc) \
|
|
if((Flags)&CFGFLAG_SERVER && !((Flags)&CFGFLAG_NONTEEHISTORIC) && pGameInfo->m_pConfig->m_##Name != (Def)) \
|
|
{ \
|
|
str_format(aJson, sizeof(aJson), "%s\"%s\":\"%d\"", \
|
|
First ? "" : ",", \
|
|
E(aBuffer1, #ScriptName), \
|
|
pGameInfo->m_pConfig->m_##Name); \
|
|
Write(aJson, str_length(aJson)); \
|
|
First = false; \
|
|
}
|
|
|
|
#define MACRO_CONFIG_STR(Name,ScriptName,Len,Def,Flags,Desc) \
|
|
if((Flags)&CFGFLAG_SERVER && !((Flags)&CFGFLAG_NONTEEHISTORIC) && str_comp(pGameInfo->m_pConfig->m_##Name, (Def)) != 0) \
|
|
{ \
|
|
str_format(aJson, sizeof(aJson), "%s\"%s\":\"%s\"", \
|
|
First ? "" : ",", \
|
|
E(aBuffer1, #ScriptName), \
|
|
E(aBuffer2, pGameInfo->m_pConfig->m_##Name)); \
|
|
Write(aJson, str_length(aJson)); \
|
|
First = false; \
|
|
}
|
|
|
|
#include <engine/shared/config_variables.h>
|
|
|
|
#undef MACRO_CONFIG_INT
|
|
#undef MACRO_CONFIG_STR
|
|
|
|
str_format(aJson, sizeof(aJson), "},\"tuning\":{");
|
|
Write(aJson, str_length(aJson));
|
|
|
|
First = true;
|
|
|
|
static const float TicksPerSecond = 50.0f;
|
|
#define MACRO_TUNING_PARAM(Name,ScriptName,Value,Description) \
|
|
if(pGameInfo->m_pTuning->m_##Name.Get() != (int)((Value)*100)) \
|
|
{ \
|
|
str_format(aJson, sizeof(aJson), "%s\"%s\":\"%d\"", \
|
|
First ? "" : ",", \
|
|
E(aBuffer1, #ScriptName), \
|
|
pGameInfo->m_pTuning->m_##Name.Get()); \
|
|
Write(aJson, str_length(aJson)); \
|
|
First = false; \
|
|
}
|
|
#include <game/tuning.h>
|
|
#undef MACRO_TUNING_PARAM
|
|
|
|
str_format(aJson, sizeof(aJson), "}}");
|
|
Write(aJson, str_length(aJson));
|
|
Write("", 1); // Null termination.
|
|
}
|
|
|
|
|
|
void CTeeHistorian::BeginTick(int Tick)
|
|
{
|
|
dbg_assert(m_State == STATE_START || m_State == STATE_BEFORE_TICK, "invalid teehistorian state");
|
|
|
|
m_Tick = Tick;
|
|
m_TickWritten = false;
|
|
|
|
if(m_Debug > 1)
|
|
{
|
|
dbg_msg("teehistorian", "tick %d", Tick);
|
|
}
|
|
|
|
m_State = STATE_BEFORE_PLAYERS;
|
|
}
|
|
|
|
void CTeeHistorian::BeginPlayers()
|
|
{
|
|
dbg_assert(m_State == STATE_BEFORE_PLAYERS, "invalid teehistorian state");
|
|
|
|
m_PrevMaxClientID = m_MaxClientID;
|
|
m_MaxClientID = -1;
|
|
|
|
m_State = STATE_PLAYERS;
|
|
}
|
|
|
|
void CTeeHistorian::EnsureTickWrittenPlayerData(int ClientID)
|
|
{
|
|
dbg_assert(ClientID > m_MaxClientID, "invalid player data order");
|
|
m_MaxClientID = ClientID;
|
|
|
|
if(!m_TickWritten && (ClientID > m_PrevMaxClientID || m_LastWrittenTick + 1 != m_Tick))
|
|
{
|
|
WriteTick();
|
|
}
|
|
else
|
|
{
|
|
// Tick is implicit.
|
|
m_LastWrittenTick = m_Tick;
|
|
m_TickWritten = true;
|
|
}
|
|
}
|
|
|
|
void CTeeHistorian::RecordPlayer(int ClientID, const CNetObj_CharacterCore *pChar)
|
|
{
|
|
dbg_assert(m_State == STATE_PLAYERS, "invalid teehistorian state");
|
|
|
|
CPlayer *pPrev = &m_aPrevPlayers[ClientID];
|
|
if(!pPrev->m_Alive || pPrev->m_X != pChar->m_X || pPrev->m_Y != pChar->m_Y)
|
|
{
|
|
EnsureTickWrittenPlayerData(ClientID);
|
|
|
|
CPacker Buffer;
|
|
Buffer.Reset();
|
|
if(pPrev->m_Alive)
|
|
{
|
|
int dx = pChar->m_X - pPrev->m_X;
|
|
int dy = pChar->m_Y - pPrev->m_Y;
|
|
Buffer.AddInt(ClientID);
|
|
Buffer.AddInt(dx);
|
|
Buffer.AddInt(dy);
|
|
if(m_Debug)
|
|
{
|
|
dbg_msg("teehistorian", "diff cid=%d dx=%d dy=%d", ClientID, dx, dy);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int x = pChar->m_X;
|
|
int y = pChar->m_Y;
|
|
Buffer.AddInt(-TEEHISTORIAN_PLAYER_NEW);
|
|
Buffer.AddInt(ClientID);
|
|
Buffer.AddInt(x);
|
|
Buffer.AddInt(y);
|
|
if(m_Debug)
|
|
{
|
|
dbg_msg("teehistorian", "new cid=%d x=%d y=%d", ClientID, x, y);
|
|
}
|
|
}
|
|
Write(Buffer.Data(), Buffer.Size());
|
|
}
|
|
pPrev->m_X = pChar->m_X;
|
|
pPrev->m_Y = pChar->m_Y;
|
|
pPrev->m_Alive = true;
|
|
}
|
|
|
|
void CTeeHistorian::RecordDeadPlayer(int ClientID)
|
|
{
|
|
dbg_assert(m_State == STATE_PLAYERS, "invalid teehistorian state");
|
|
|
|
CPlayer *pPrev = &m_aPrevPlayers[ClientID];
|
|
if(pPrev->m_Alive)
|
|
{
|
|
EnsureTickWrittenPlayerData(ClientID);
|
|
|
|
CPacker Buffer;
|
|
Buffer.Reset();
|
|
Buffer.AddInt(-TEEHISTORIAN_PLAYER_OLD);
|
|
Buffer.AddInt(ClientID);
|
|
if(m_Debug)
|
|
{
|
|
dbg_msg("teehistorian", "old cid=%d", ClientID);
|
|
}
|
|
Write(Buffer.Data(), Buffer.Size());
|
|
}
|
|
pPrev->m_Alive = false;
|
|
}
|
|
|
|
void CTeeHistorian::Write(const void *pData, int DataSize)
|
|
{
|
|
m_pfnWriteCallback(pData, DataSize, m_pWriteCallbackUserdata);
|
|
}
|
|
|
|
void CTeeHistorian::EnsureTickWritten()
|
|
{
|
|
if(!m_TickWritten)
|
|
{
|
|
WriteTick();
|
|
}
|
|
}
|
|
|
|
void CTeeHistorian::WriteTick()
|
|
{
|
|
CPacker TickPacker;
|
|
TickPacker.Reset();
|
|
|
|
int dt = m_Tick - m_LastWrittenTick - 1;
|
|
TickPacker.AddInt(-TEEHISTORIAN_TICK_SKIP);
|
|
TickPacker.AddInt(dt);
|
|
if(m_Debug)
|
|
{
|
|
dbg_msg("teehistorian", "skip_ticks dt=%d", dt);
|
|
}
|
|
Write(TickPacker.Data(), TickPacker.Size());
|
|
|
|
m_TickWritten = true;
|
|
m_LastWrittenTick = m_Tick;
|
|
}
|
|
|
|
void CTeeHistorian::EndPlayers()
|
|
{
|
|
dbg_assert(m_State == STATE_PLAYERS, "invalid teehistorian state");
|
|
|
|
m_State = STATE_BEFORE_INPUTS;
|
|
}
|
|
|
|
void CTeeHistorian::BeginInputs()
|
|
{
|
|
dbg_assert(m_State == STATE_BEFORE_INPUTS, "invalid teehistorian state");
|
|
|
|
m_State = STATE_INPUTS;
|
|
}
|
|
|
|
void CTeeHistorian::RecordPlayerInput(int ClientID, const CNetObj_PlayerInput *pInput)
|
|
{
|
|
CPacker Buffer;
|
|
|
|
CPlayer *pPrev = &m_aPrevPlayers[ClientID];
|
|
CNetObj_PlayerInput DiffInput;
|
|
if(pPrev->m_InputExists)
|
|
{
|
|
if(mem_comp(&pPrev->m_Input, pInput, sizeof(pPrev->m_Input)) == 0)
|
|
{
|
|
return;
|
|
}
|
|
EnsureTickWritten();
|
|
Buffer.Reset();
|
|
|
|
Buffer.AddInt(-TEEHISTORIAN_INPUT_DIFF);
|
|
CSnapshotDelta::DiffItem((int *)&pPrev->m_Input, (int *)pInput, (int *)&DiffInput, sizeof(DiffInput) / sizeof(int));
|
|
if(m_Debug)
|
|
{
|
|
const int *pData = (const int *)&DiffInput;
|
|
dbg_msg("teehistorian", "diff_input cid=%d %d %d %d %d %d %d %d %d %d %d", ClientID,
|
|
pData[0], pData[1], pData[2], pData[3], pData[4],
|
|
pData[5], pData[6], pData[7], pData[8], pData[9]);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
EnsureTickWritten();
|
|
Buffer.Reset();
|
|
Buffer.AddInt(-TEEHISTORIAN_INPUT_NEW);
|
|
DiffInput = *pInput;
|
|
if(m_Debug)
|
|
{
|
|
dbg_msg("teehistorian", "new_input cid=%d", ClientID);
|
|
}
|
|
}
|
|
Buffer.AddInt(ClientID);
|
|
for(int i = 0; i < (int)(sizeof(DiffInput) / sizeof(int)); i++)
|
|
{
|
|
Buffer.AddInt(((int *)&DiffInput)[i]);
|
|
}
|
|
pPrev->m_InputExists = true;
|
|
pPrev->m_Input = *pInput;
|
|
|
|
Write(Buffer.Data(), Buffer.Size());
|
|
}
|
|
|
|
void CTeeHistorian::RecordPlayerMessage(int ClientID, const void *pMsg, int MsgSize)
|
|
{
|
|
EnsureTickWritten();
|
|
|
|
CPacker Buffer;
|
|
Buffer.Reset();
|
|
Buffer.AddInt(-TEEHISTORIAN_MESSAGE);
|
|
Buffer.AddInt(ClientID);
|
|
Buffer.AddInt(MsgSize);
|
|
Buffer.AddRaw(pMsg, MsgSize);
|
|
|
|
if(m_Debug)
|
|
{
|
|
CUnpacker Unpacker;
|
|
Unpacker.Reset(pMsg, MsgSize);
|
|
int MsgID = Unpacker.GetInt();
|
|
int Sys = MsgID & 1;
|
|
MsgID >>= 1;
|
|
dbg_msg("teehistorian", "msg cid=%d sys=%d msgid=%d", ClientID, Sys, MsgID);
|
|
}
|
|
|
|
Write(Buffer.Data(), Buffer.Size());
|
|
}
|
|
|
|
void CTeeHistorian::RecordPlayerJoin(int ClientID)
|
|
{
|
|
EnsureTickWritten();
|
|
|
|
CPacker Buffer;
|
|
Buffer.Reset();
|
|
Buffer.AddInt(-TEEHISTORIAN_JOIN);
|
|
Buffer.AddInt(ClientID);
|
|
|
|
if(m_Debug)
|
|
{
|
|
dbg_msg("teehistorian", "join cid=%d", ClientID);
|
|
}
|
|
|
|
Write(Buffer.Data(), Buffer.Size());
|
|
}
|
|
|
|
void CTeeHistorian::RecordPlayerDrop(int ClientID, const char *pReason)
|
|
{
|
|
EnsureTickWritten();
|
|
|
|
CPacker Buffer;
|
|
Buffer.Reset();
|
|
Buffer.AddInt(-TEEHISTORIAN_DROP);
|
|
Buffer.AddInt(ClientID);
|
|
Buffer.AddString(pReason, 0);
|
|
|
|
if(m_Debug)
|
|
{
|
|
dbg_msg("teehistorian", "drop cid=%d reason='%s'", ClientID, pReason);
|
|
}
|
|
|
|
Write(Buffer.Data(), Buffer.Size());
|
|
}
|
|
|
|
void CTeeHistorian::RecordConsoleCommand(int ClientID, int FlagMask, const char *pCmd, IConsole::IResult *pResult)
|
|
{
|
|
EnsureTickWritten();
|
|
|
|
CPacker Buffer;
|
|
Buffer.Reset();
|
|
Buffer.AddInt(-TEEHISTORIAN_CONSOLE_COMMAND);
|
|
Buffer.AddInt(ClientID);
|
|
Buffer.AddInt(FlagMask);
|
|
Buffer.AddString(pCmd, 0);
|
|
Buffer.AddInt(pResult->NumArguments());
|
|
for(int i = 0; i < pResult->NumArguments(); i++)
|
|
{
|
|
Buffer.AddString(pResult->GetString(i), 0);
|
|
}
|
|
|
|
if(m_Debug)
|
|
{
|
|
dbg_msg("teehistorian", "ccmd cid=%d cmd='%s'", ClientID, pCmd);
|
|
}
|
|
|
|
Write(Buffer.Data(), Buffer.Size());
|
|
}
|
|
|
|
void CTeeHistorian::EndInputs()
|
|
{
|
|
dbg_assert(m_State == STATE_INPUTS, "invalid teehistorian state");
|
|
|
|
m_State = STATE_BEFORE_ENDTICK;
|
|
}
|
|
|
|
void CTeeHistorian::EndTick()
|
|
{
|
|
dbg_assert(m_State == STATE_BEFORE_ENDTICK, "invalid teehistorian state");
|
|
m_State = STATE_BEFORE_TICK;
|
|
}
|
|
|
|
void CTeeHistorian::Finish()
|
|
{
|
|
dbg_assert(m_State == STATE_START || m_State == STATE_INPUTS || m_State == STATE_BEFORE_ENDTICK || m_State == STATE_BEFORE_TICK, "invalid teehistorian state");
|
|
|
|
if(m_State == STATE_INPUTS)
|
|
{
|
|
EndInputs();
|
|
}
|
|
if(m_State == STATE_BEFORE_ENDTICK)
|
|
{
|
|
EndTick();
|
|
}
|
|
|
|
CPacker Buffer;
|
|
Buffer.Reset();
|
|
Buffer.AddInt(-TEEHISTORIAN_FINISH);
|
|
|
|
if(m_Debug)
|
|
{
|
|
dbg_msg("teehistorian", "finish");
|
|
}
|
|
|
|
Write(Buffer.Data(), Buffer.Size());
|
|
}
|