ddnet/src/engine/shared/masterserver.cpp
2022-05-17 20:33:29 +02:00

232 lines
4.8 KiB
C++

/* (c) Magnus Auvinen. See licence.txt in the root of the distribution for more information. */
/* If you are missing that file, acquire a complete release at teeworlds.com. */
#include <cstdio> // sscanf
#include <base/system.h>
#include <engine/engine.h>
#include <engine/masterserver.h>
#include <engine/storage.h>
#include "linereader.h"
class CMasterServer : public IEngineMasterServer
{
public:
// master server functions
struct CMasterInfo
{
char m_aHostname[128];
NETADDR m_Addr;
bool m_Valid;
int m_Count;
std::shared_ptr<CHostLookup> m_pLookup;
};
enum
{
STATE_INIT,
STATE_UPDATE,
STATE_READY,
};
CMasterInfo m_aMasterServers[MAX_MASTERSERVERS];
std::shared_ptr<CHostLookup> m_apLookup[MAX_MASTERSERVERS];
int m_State;
IEngine *m_pEngine;
IStorage *m_pStorage;
CMasterServer()
{
SetDefault();
m_State = STATE_INIT;
m_pEngine = 0;
m_pStorage = 0;
}
int RefreshAddresses(int Nettype) override
{
if(m_State != STATE_INIT && m_State != STATE_READY)
return -1;
dbg_msg("engine/mastersrv", "refreshing master server addresses");
// add lookup jobs
for(int i = 0; i < MAX_MASTERSERVERS; i++)
{
*m_apLookup[i] = CHostLookup(m_aMasterServers[i].m_aHostname, Nettype);
m_pEngine->AddJob(m_apLookup[i]);
m_aMasterServers[i].m_Valid = false;
m_aMasterServers[i].m_Count = 0;
}
m_State = STATE_UPDATE;
return 0;
}
void Update() override
{
// check if we need to update
if(m_State != STATE_UPDATE)
return;
m_State = STATE_READY;
for(int i = 0; i < MAX_MASTERSERVERS; i++)
{
if(m_apLookup[i]->Status() != IJob::STATE_DONE)
m_State = STATE_UPDATE;
else
{
if(m_apLookup[i]->m_Result == 0)
{
m_aMasterServers[i].m_Addr = m_apLookup[i]->m_Addr;
m_aMasterServers[i].m_Addr.port = 8300;
m_aMasterServers[i].m_Valid = true;
}
else
{
m_aMasterServers[i].m_Valid = false;
}
}
}
if(m_State == STATE_READY)
{
dbg_msg("engine/mastersrv", "saving addresses");
Save();
}
}
bool IsRefreshing() const override
{
return m_State != STATE_READY;
}
NETADDR GetAddr(int Index) const override
{
return m_aMasterServers[Index].m_Addr;
}
void SetCount(int Index, int Count) override
{
m_aMasterServers[Index].m_Count = Count;
}
int GetCount(int Index) const override
{
return m_aMasterServers[Index].m_Count;
}
const char *GetName(int Index) const override
{
return m_aMasterServers[Index].m_aHostname;
}
bool IsValid(int Index) const override
{
return m_aMasterServers[Index].m_Valid;
}
void Init() override
{
m_pEngine = Kernel()->RequestInterface<IEngine>();
m_pStorage = Kernel()->RequestInterface<IStorage>();
}
void SetDefault() override
{
mem_zero(m_aMasterServers, sizeof(m_aMasterServers));
for(int i = 0; i < MAX_MASTERSERVERS; i++)
{
str_format(m_aMasterServers[i].m_aHostname, sizeof(m_aMasterServers[i].m_aHostname), "master%d.teeworlds.com", i + 1);
m_apLookup[i] = std::make_shared<CHostLookup>();
}
}
int Load() override
{
if(!m_pStorage)
return -1;
// try to open file
IOHANDLE File = m_pStorage->OpenFile("masters.cfg", IOFLAG_READ | IOFLAG_SKIP_BOM, IStorage::TYPE_SAVE);
if(!File)
return -1;
CLineReader LineReader;
LineReader.Init(File);
while(true)
{
CMasterInfo Info = {{0}};
const char *pLine = LineReader.Get();
if(!pLine)
break;
// parse line
char aAddrStr[NETADDR_MAXSTRSIZE];
if(sscanf(pLine, "%127s %47s", Info.m_aHostname, aAddrStr) == 2 && net_addr_from_str(&Info.m_Addr, aAddrStr) == 0)
{
Info.m_Addr.port = 8300;
bool Added = false;
for(auto &MasterServer : m_aMasterServers)
{
if(str_comp(MasterServer.m_aHostname, Info.m_aHostname) == 0)
{
MasterServer = Info;
Added = true;
break;
}
}
if(!Added)
{
for(auto &MasterServer : m_aMasterServers)
{
if(MasterServer.m_Addr.type == NETTYPE_INVALID)
{
MasterServer = Info;
Added = true;
break;
}
}
}
if(!Added)
break;
}
}
io_close(File);
return 0;
}
int Save() override
{
if(!m_pStorage)
return -1;
// try to open file
IOHANDLE File = m_pStorage->OpenFile("masters.cfg", IOFLAG_WRITE, IStorage::TYPE_SAVE);
if(!File)
return -1;
for(auto &MasterServer : m_aMasterServers)
{
char aAddrStr[NETADDR_MAXSTRSIZE];
if(MasterServer.m_Addr.type != NETTYPE_INVALID)
net_addr_str(&MasterServer.m_Addr, aAddrStr, sizeof(aAddrStr), true);
else
aAddrStr[0] = 0;
char aBuf[256];
str_format(aBuf, sizeof(aBuf), "%s %s", MasterServer.m_aHostname, aAddrStr);
io_write(File, aBuf, str_length(aBuf));
io_write_newline(File);
}
io_close(File);
return 0;
}
};
IEngineMasterServer *CreateEngineMasterServer() { return new CMasterServer; }