mirror of
https://github.com/ddnet/ddnet.git
synced 2024-11-11 02:28:18 +00:00
6600024f24
Support is incomplete for `leak_ip_address_to_all_servers` (will only ping the first address of each server) and for the `leak_ip` setting (which will also only ping the first address of each server).
168 lines
4.5 KiB
C++
168 lines
4.5 KiB
C++
#include "serverbrowser_ping_cache.h"
|
|
|
|
#include <engine/console.h>
|
|
#include <engine/sqlite.h>
|
|
|
|
#include <sqlite3.h>
|
|
|
|
#include <unordered_map>
|
|
#include <vector>
|
|
|
|
class CServerBrowserPingCache : public IServerBrowserPingCache
|
|
{
|
|
public:
|
|
class CEntry
|
|
{
|
|
public:
|
|
NETADDR m_Addr;
|
|
int m_Ping;
|
|
};
|
|
|
|
CServerBrowserPingCache(IConsole *pConsole, IStorage *pStorage);
|
|
virtual ~CServerBrowserPingCache() = default;
|
|
|
|
void Load() override;
|
|
|
|
int NumEntries() const override;
|
|
void CachePing(const NETADDR &Addr, int Ping) override;
|
|
int GetPing(const NETADDR *pAddrs, int NumAddrs) const override;
|
|
|
|
private:
|
|
IConsole *m_pConsole;
|
|
|
|
CSqlite m_pDisk;
|
|
CSqliteStmt m_pLoadStmt;
|
|
CSqliteStmt m_pStoreStmt;
|
|
|
|
std::unordered_map<NETADDR, int> m_Entries;
|
|
};
|
|
|
|
CServerBrowserPingCache::CServerBrowserPingCache(IConsole *pConsole, IStorage *pStorage) :
|
|
m_pConsole(pConsole)
|
|
{
|
|
m_pDisk = SqliteOpen(pConsole, pStorage, "ddnet-cache.sqlite3");
|
|
if(!m_pDisk)
|
|
{
|
|
pConsole->Print(IConsole::OUTPUT_LEVEL_STANDARD, "serverbrowse_ping_cache", "failed to open ddnet-cache.sqlite3");
|
|
return;
|
|
}
|
|
sqlite3 *pSqlite = m_pDisk.get();
|
|
static const char TABLE[] = "CREATE TABLE IF NOT EXISTS server_pings (ip_address TEXT PRIMARY KEY NOT NULL, ping INTEGER NOT NULL, utc_timestamp TEXT NOT NULL)";
|
|
if(SQLITE_HANDLE_ERROR(sqlite3_exec(pSqlite, TABLE, nullptr, nullptr, nullptr)))
|
|
{
|
|
m_pDisk = nullptr;
|
|
pConsole->Print(IConsole::OUTPUT_LEVEL_STANDARD, "serverbrowse_ping_cache", "failed to create server_pings table");
|
|
return;
|
|
}
|
|
m_pLoadStmt = SqlitePrepare(pConsole, pSqlite, "SELECT ip_address, ping FROM server_pings");
|
|
m_pStoreStmt = SqlitePrepare(pConsole, pSqlite, "INSERT OR REPLACE INTO server_pings (ip_address, ping, utc_timestamp) VALUES (?, ?, datetime('now'))");
|
|
}
|
|
|
|
void CServerBrowserPingCache::Load()
|
|
{
|
|
if(m_pDisk)
|
|
{
|
|
std::vector<CEntry> vNewEntries;
|
|
|
|
sqlite3 *pSqlite = m_pDisk.get();
|
|
IConsole *pConsole = m_pConsole;
|
|
bool Error = false;
|
|
bool WarnedForBadAddress = false;
|
|
Error = Error || !m_pLoadStmt;
|
|
while(!Error)
|
|
{
|
|
int StepResult = SQLITE_HANDLE_ERROR(sqlite3_step(m_pLoadStmt.get()));
|
|
if(StepResult == SQLITE_DONE)
|
|
{
|
|
break;
|
|
}
|
|
else if(StepResult == SQLITE_ROW)
|
|
{
|
|
const char *pIpAddress = (const char *)sqlite3_column_text(m_pLoadStmt.get(), 0);
|
|
int Ping = sqlite3_column_int(m_pLoadStmt.get(), 1);
|
|
NETADDR Addr;
|
|
if(net_addr_from_str(&Addr, pIpAddress))
|
|
{
|
|
if(!WarnedForBadAddress)
|
|
{
|
|
char aBuf[64];
|
|
str_format(aBuf, sizeof(aBuf), "invalid address: %s", pIpAddress);
|
|
pConsole->Print(IConsole::OUTPUT_LEVEL_STANDARD, "serverbrowse_ping_cache", aBuf);
|
|
WarnedForBadAddress = true;
|
|
}
|
|
continue;
|
|
}
|
|
vNewEntries.push_back(CEntry{Addr, Ping});
|
|
}
|
|
else
|
|
{
|
|
Error = true;
|
|
}
|
|
}
|
|
if(Error)
|
|
{
|
|
pConsole->Print(IConsole::OUTPUT_LEVEL_STANDARD, "serverbrowse_ping_cache", "failed to load ping cache");
|
|
return;
|
|
}
|
|
for(const auto &Entry : vNewEntries)
|
|
{
|
|
m_Entries[Entry.m_Addr] = Entry.m_Ping;
|
|
}
|
|
}
|
|
}
|
|
|
|
int CServerBrowserPingCache::NumEntries() const
|
|
{
|
|
return m_Entries.size();
|
|
}
|
|
|
|
void CServerBrowserPingCache::CachePing(const NETADDR &Addr, int Ping)
|
|
{
|
|
NETADDR AddrWithoutPort = Addr;
|
|
AddrWithoutPort.port = 0;
|
|
m_Entries[AddrWithoutPort] = Ping;
|
|
if(m_pDisk)
|
|
{
|
|
sqlite3 *pSqlite = m_pDisk.get();
|
|
IConsole *pConsole = m_pConsole;
|
|
char aAddr[NETADDR_MAXSTRSIZE];
|
|
net_addr_str(&AddrWithoutPort, aAddr, sizeof(aAddr), false);
|
|
|
|
bool Error = false;
|
|
Error = Error || !m_pStoreStmt;
|
|
Error = Error || SQLITE_HANDLE_ERROR(sqlite3_reset(m_pStoreStmt.get())) != SQLITE_OK;
|
|
Error = Error || SQLITE_HANDLE_ERROR(sqlite3_bind_text(m_pStoreStmt.get(), 1, aAddr, -1, SQLITE_STATIC)) != SQLITE_OK;
|
|
Error = Error || SQLITE_HANDLE_ERROR(sqlite3_bind_int(m_pStoreStmt.get(), 2, Ping)) != SQLITE_OK;
|
|
Error = Error || SQLITE_HANDLE_ERROR(sqlite3_step(m_pStoreStmt.get())) != SQLITE_DONE;
|
|
if(Error)
|
|
{
|
|
pConsole->Print(IConsole::OUTPUT_LEVEL_STANDARD, "serverbrowse_ping_cache", "failed to store ping");
|
|
}
|
|
}
|
|
}
|
|
|
|
int CServerBrowserPingCache::GetPing(const NETADDR *pAddrs, int NumAddrs) const
|
|
{
|
|
int Ping = -1;
|
|
for(int i = 0; i < NumAddrs; i++)
|
|
{
|
|
NETADDR Addr = pAddrs[i];
|
|
Addr.port = 0;
|
|
auto Entry = m_Entries.find(Addr);
|
|
if(Entry == m_Entries.end())
|
|
{
|
|
continue;
|
|
}
|
|
if(Ping == -1 || Entry->second < Ping)
|
|
{
|
|
Ping = Entry->second;
|
|
}
|
|
}
|
|
return Ping;
|
|
}
|
|
|
|
IServerBrowserPingCache *CreateServerBrowserPingCache(IConsole *pConsole, IStorage *pStorage)
|
|
{
|
|
return new CServerBrowserPingCache(pConsole, pStorage);
|
|
}
|