2022-02-27 07:53:44 -05:00
|
|
|
#include <STDInclude.hpp>
|
2023-01-03 07:16:44 -05:00
|
|
|
#include <Utils/Compression.hpp>
|
|
|
|
#include <Utils/InfoString.hpp>
|
|
|
|
|
|
|
|
#include <proto/node.pb.h>
|
|
|
|
|
2023-04-17 08:47:29 -04:00
|
|
|
#include "Node.hpp"
|
2022-12-26 07:07:24 -05:00
|
|
|
#include "ServerList.hpp"
|
|
|
|
#include "Session.hpp"
|
2018-10-09 04:53:15 -04:00
|
|
|
|
|
|
|
namespace Components
|
|
|
|
{
|
|
|
|
std::recursive_mutex Node::Mutex;
|
|
|
|
std::vector<Node::Entry> Node::Nodes;
|
|
|
|
|
2023-05-06 09:20:56 -04:00
|
|
|
bool Node::WasIngame = false;
|
2018-10-09 04:53:15 -04:00
|
|
|
|
2023-05-14 12:42:42 -04:00
|
|
|
bool Node::Entry::isValid() const
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
|
|
|
return (this->lastResponse.has_value() && !this->lastResponse->elapsed(NODE_HALFLIFE * 2));
|
|
|
|
}
|
|
|
|
|
2023-05-14 12:42:42 -04:00
|
|
|
bool Node::Entry::isDead() const
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
|
|
|
if (!this->lastResponse.has_value())
|
|
|
|
{
|
|
|
|
if (this->lastRequest.has_value() && this->lastRequest->elapsed(NODE_HALFLIFE))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
2022-06-25 12:21:12 -04:00
|
|
|
else if (this->lastResponse->elapsed(NODE_HALFLIFE * 2) && this->lastRequest.has_value() && this->lastRequest->after(*this->lastResponse))
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-05-14 12:42:42 -04:00
|
|
|
bool Node::Entry::requiresRequest() const
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
|
|
|
return (!this->isDead() && (!this->lastRequest.has_value() || this->lastRequest->elapsed(NODE_HALFLIFE)));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Node::Entry::sendRequest()
|
|
|
|
{
|
|
|
|
if (!this->lastRequest.has_value()) this->lastRequest.emplace(Utils::Time::Point());
|
|
|
|
this->lastRequest->update();
|
|
|
|
|
|
|
|
Session::Send(this->address, "nodeListRequest");
|
|
|
|
Node::SendList(this->address);
|
2023-05-14 12:42:42 -04:00
|
|
|
#ifdef NODE_SYSTEM_DEBUG
|
2022-08-19 19:10:35 -04:00
|
|
|
Logger::Debug("Sent request to {}", this->address.getString());
|
2023-05-14 12:42:42 -04:00
|
|
|
#endif
|
2018-10-09 04:53:15 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Node::Entry::reset()
|
|
|
|
{
|
|
|
|
// this->lastResponse.reset(); // This would invalidate the node, but maybe we don't want that?
|
|
|
|
this->lastRequest.reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Node::LoadNodePreset()
|
|
|
|
{
|
|
|
|
Proto::Node::List list;
|
|
|
|
|
2022-10-15 16:31:16 -04:00
|
|
|
FileSystem::File defaultNodes("nodes_default.dat");
|
|
|
|
if (!defaultNodes.exists() || !list.ParseFromString(Utils::Compression::ZLib::Decompress(defaultNodes.getBuffer()))) return;
|
2018-10-09 04:53:15 -04:00
|
|
|
|
2022-10-15 16:31:16 -04:00
|
|
|
for (auto i = 0; i < list.nodes_size(); ++i)
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
|
|
|
const std::string& addr = list.nodes(i);
|
|
|
|
|
|
|
|
if (addr.size() == sizeof(sockaddr))
|
|
|
|
{
|
|
|
|
Node::Add(reinterpret_cast<sockaddr*>(const_cast<char*>(addr.data())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Node::LoadNodes()
|
|
|
|
{
|
|
|
|
Proto::Node::List list;
|
|
|
|
std::string nodes = Utils::IO::ReadFile("players/nodes.dat");
|
|
|
|
if (nodes.empty() || !list.ParseFromString(Utils::Compression::ZLib::Decompress(nodes))) return;
|
|
|
|
|
|
|
|
for (int i = 0; i < list.nodes_size(); ++i)
|
|
|
|
{
|
|
|
|
const std::string& addr = list.nodes(i);
|
|
|
|
|
|
|
|
if (addr.size() == sizeof(sockaddr))
|
|
|
|
{
|
|
|
|
Node::Add(reinterpret_cast<sockaddr*>(const_cast<char*>(addr.data())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Node::StoreNodes(bool force)
|
|
|
|
{
|
2022-05-03 19:03:11 -04:00
|
|
|
if (Dedicated::IsEnabled() && Dedicated::SVLanOnly.get<bool>()) return;
|
2018-10-09 04:53:15 -04:00
|
|
|
|
|
|
|
static Utils::Time::Interval interval;
|
|
|
|
if (!force && !interval.elapsed(1min)) return;
|
|
|
|
interval.update();
|
|
|
|
|
|
|
|
Proto::Node::List list;
|
|
|
|
|
|
|
|
Node::Mutex.lock();
|
|
|
|
for (auto& node : Node::Nodes)
|
|
|
|
{
|
|
|
|
if (node.isValid())
|
|
|
|
{
|
|
|
|
std::string* str = list.add_nodes();
|
|
|
|
|
|
|
|
sockaddr addr = node.address.getSockAddr();
|
|
|
|
str->append(reinterpret_cast<char*>(&addr), sizeof(addr));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Node::Mutex.unlock();
|
|
|
|
|
|
|
|
Utils::IO::WriteFile("players/nodes.dat", Utils::Compression::ZLib::Compress(list.SerializeAsString()));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Node::Add(Network::Address address)
|
|
|
|
{
|
|
|
|
#ifndef DEBUG
|
|
|
|
if (address.isLocal() || address.isSelf()) return;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!address.isValid()) return;
|
|
|
|
|
2022-06-16 10:15:26 -04:00
|
|
|
std::lock_guard _(Node::Mutex);
|
2018-10-09 04:53:15 -04:00
|
|
|
for (auto& session : Node::Nodes)
|
|
|
|
{
|
|
|
|
if (session.address == address) return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node::Entry node;
|
|
|
|
node.address = address;
|
|
|
|
|
|
|
|
Node::Nodes.push_back(node);
|
|
|
|
}
|
|
|
|
|
2018-12-02 12:17:45 -05:00
|
|
|
std::vector<Node::Entry> Node::GetNodes()
|
|
|
|
{
|
2022-06-16 10:15:26 -04:00
|
|
|
std::lock_guard _(Node::Mutex);
|
2018-12-02 12:17:45 -05:00
|
|
|
|
|
|
|
return Node::Nodes;
|
|
|
|
}
|
|
|
|
|
2018-10-09 04:53:15 -04:00
|
|
|
void Node::RunFrame()
|
|
|
|
{
|
2022-05-03 19:03:11 -04:00
|
|
|
if (Dedicated::IsEnabled() && Dedicated::SVLanOnly.get<bool>()) return;
|
2018-10-09 04:53:15 -04:00
|
|
|
|
2022-05-10 07:22:36 -04:00
|
|
|
if (!Dedicated::IsEnabled())
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
2022-10-15 16:31:16 -04:00
|
|
|
if (ServerList::UseMasterServer) return; // don't run node frame if master server is active
|
2022-05-10 07:22:36 -04:00
|
|
|
|
2023-05-06 09:20:56 -04:00
|
|
|
if (Game::CL_GetLocalClientConnectionState(0) != Game::CA_DISCONNECTED)
|
2022-05-10 07:22:36 -04:00
|
|
|
{
|
2023-05-06 09:20:56 -04:00
|
|
|
WasIngame = true;
|
|
|
|
return; // don't run while in-game because it can still cause lag spikes on lower end PCs
|
2022-05-10 07:22:36 -04:00
|
|
|
}
|
2018-10-09 04:53:15 -04:00
|
|
|
}
|
|
|
|
|
2023-05-06 09:20:56 -04:00
|
|
|
if (WasIngame) // our last frame we were in-game and now we aren't so touch all nodes
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
|
|
|
for (auto i = Node::Nodes.begin(); i != Node::Nodes.end();++i)
|
|
|
|
{
|
|
|
|
// clearing the last request and response times makes the
|
|
|
|
// dispatcher think its a new node and will force a refresh
|
|
|
|
i->lastRequest.reset();
|
|
|
|
i->lastResponse.reset();
|
|
|
|
}
|
2023-05-06 09:20:56 -04:00
|
|
|
|
|
|
|
WasIngame = false;
|
2018-10-09 04:53:15 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static Utils::Time::Interval frameLimit;
|
|
|
|
int interval = static_cast<int>(1000.0f / Dvar::Var("net_serverFrames").get<int>());
|
|
|
|
if (!frameLimit.elapsed(std::chrono::milliseconds(interval))) return;
|
|
|
|
frameLimit.update();
|
|
|
|
|
2022-06-16 10:15:26 -04:00
|
|
|
std::lock_guard _(Node::Mutex);
|
2018-10-09 04:53:15 -04:00
|
|
|
Dvar::Var queryLimit("net_serverQueryLimit");
|
|
|
|
|
|
|
|
int sentRequests = 0;
|
|
|
|
for (auto i = Node::Nodes.begin(); i != Node::Nodes.end();)
|
|
|
|
{
|
|
|
|
if (i->isDead())
|
|
|
|
{
|
|
|
|
i = Node::Nodes.erase(i);
|
|
|
|
continue;
|
|
|
|
}
|
2022-06-25 12:21:12 -04:00
|
|
|
if (sentRequests < queryLimit.get<int>() && i->requiresRequest())
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
|
|
|
++sentRequests;
|
|
|
|
i->sendRequest();
|
|
|
|
}
|
|
|
|
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Node::Synchronize()
|
|
|
|
{
|
2022-06-16 10:15:26 -04:00
|
|
|
std::lock_guard _(Node::Mutex);
|
2018-10-09 04:53:15 -04:00
|
|
|
for (auto& node : Node::Nodes)
|
|
|
|
{
|
|
|
|
//if (node.isValid()) // Comment out to simulate 'syncnodes' behaviour
|
|
|
|
{
|
|
|
|
node.reset();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-17 08:29:18 -05:00
|
|
|
void Node::HandleResponse(Network::Address address, const std::string& data)
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
|
|
|
Proto::Node::List list;
|
|
|
|
if (!list.ParseFromString(data)) return;
|
|
|
|
|
2023-05-14 12:42:42 -04:00
|
|
|
#ifdef NODE_SYSTEM_DEBUG
|
2022-08-19 19:10:35 -04:00
|
|
|
Logger::Debug("Received response from {}", address.getString());
|
2023-05-14 12:42:42 -04:00
|
|
|
#endif
|
2018-10-09 04:53:15 -04:00
|
|
|
|
2022-06-16 10:15:26 -04:00
|
|
|
std::lock_guard _(Node::Mutex);
|
2018-10-09 04:53:15 -04:00
|
|
|
|
|
|
|
for (int i = 0; i < list.nodes_size(); ++i)
|
|
|
|
{
|
|
|
|
const std::string& addr = list.nodes(i);
|
|
|
|
|
|
|
|
if (addr.size() == sizeof(sockaddr))
|
|
|
|
{
|
|
|
|
Node::Add(reinterpret_cast<sockaddr*>(const_cast<char*>(addr.data())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (list.isnode() && (!list.port() || list.port() == address.getPort()))
|
|
|
|
{
|
2022-10-15 16:31:16 -04:00
|
|
|
if (!Dedicated::IsEnabled() && ServerList::IsOnlineList() && !ServerList::UseMasterServer && list.protocol() == PROTOCOL)
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
2023-05-14 12:42:42 -04:00
|
|
|
#ifdef NODE_SYSTEM_DEBUG
|
2022-08-19 19:10:35 -04:00
|
|
|
Logger::Debug("Inserting {} into the serverlist", address.getString());
|
2023-05-14 12:42:42 -04:00
|
|
|
#endif
|
2018-10-09 04:53:15 -04:00
|
|
|
ServerList::InsertRequest(address);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-05-14 12:42:42 -04:00
|
|
|
#ifdef NODE_SYSTEM_DEBUG
|
2022-08-19 19:10:35 -04:00
|
|
|
Logger::Debug("Dropping serverlist insertion for {}", address.getString());
|
2023-05-14 12:42:42 -04:00
|
|
|
#endif
|
2018-10-09 04:53:15 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& node : Node::Nodes)
|
|
|
|
{
|
|
|
|
if (address == node.address)
|
|
|
|
{
|
|
|
|
if (!node.lastResponse.has_value()) node.lastResponse.emplace(Utils::Time::Point());
|
|
|
|
node.lastResponse->update();
|
|
|
|
|
|
|
|
node.data.protocol = list.protocol();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Node::Entry entry;
|
|
|
|
entry.address = address;
|
|
|
|
entry.data.protocol = list.protocol();
|
|
|
|
entry.lastResponse.emplace(Utils::Time::Point());
|
|
|
|
|
|
|
|
Node::Nodes.push_back(entry);
|
|
|
|
}
|
|
|
|
}
|
2018-12-02 12:17:45 -05:00
|
|
|
|
2022-06-16 10:15:26 -04:00
|
|
|
void Node::SendList(const Network::Address& address)
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
2022-06-16 10:15:26 -04:00
|
|
|
std::lock_guard _(Node::Mutex);
|
2018-10-09 04:53:15 -04:00
|
|
|
|
2020-11-17 02:06:02 -05:00
|
|
|
// need to keep the message size below 1404 bytes else recipient will just drop it
|
|
|
|
std::vector<std::string> nodeListReponseMessages;
|
|
|
|
|
2022-06-04 08:59:14 -04:00
|
|
|
for (std::size_t curNode = 0; curNode < Node::Nodes.size();)
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
2020-11-17 02:06:02 -05:00
|
|
|
Proto::Node::List list;
|
|
|
|
list.set_isnode(Dedicated::IsEnabled());
|
|
|
|
list.set_protocol(PROTOCOL);
|
|
|
|
list.set_port(Node::GetPort());
|
|
|
|
|
2022-06-04 08:59:14 -04:00
|
|
|
for (std::size_t i = 0; i < NODE_MAX_NODES_TO_SEND;)
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
2020-11-17 02:06:02 -05:00
|
|
|
if (curNode >= Node::Nodes.size())
|
|
|
|
break;
|
2018-10-09 04:53:15 -04:00
|
|
|
|
2020-11-17 02:06:02 -05:00
|
|
|
auto node = Node::Nodes.at(curNode++);
|
|
|
|
|
|
|
|
if (node.isValid())
|
|
|
|
{
|
|
|
|
std::string* str = list.add_nodes();
|
|
|
|
|
|
|
|
sockaddr addr = node.address.getSockAddr();
|
|
|
|
str->append(reinterpret_cast<char*>(&addr), sizeof(addr));
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
2018-10-09 04:53:15 -04:00
|
|
|
}
|
2020-11-17 02:06:02 -05:00
|
|
|
|
|
|
|
nodeListReponseMessages.push_back(list.SerializeAsString());
|
2018-10-09 04:53:15 -04:00
|
|
|
}
|
|
|
|
|
2022-06-04 08:59:14 -04:00
|
|
|
auto i = 0;
|
|
|
|
for (const auto& nodeListData : nodeListReponseMessages)
|
2020-11-17 02:06:02 -05:00
|
|
|
{
|
2022-06-16 10:15:26 -04:00
|
|
|
Scheduler::Once([=]
|
2020-11-17 02:06:02 -05:00
|
|
|
{
|
2023-05-14 12:42:42 -04:00
|
|
|
#ifdef NODE_SYSTEM_DEBUG
|
2022-06-16 10:15:26 -04:00
|
|
|
Logger::Debug("Sending {} nodeListResponse length to {}\n", nodeListData.length(), address.getCString());
|
|
|
|
#endif
|
2020-11-17 02:06:02 -05:00
|
|
|
Session::Send(address, "nodeListResponse", nodeListData);
|
2022-06-25 12:21:12 -04:00
|
|
|
}, Scheduler::Pipeline::MAIN, NODE_SEND_RATE * i++);
|
2020-11-17 02:06:02 -05:00
|
|
|
}
|
2018-10-09 04:53:15 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned short Node::GetPort()
|
|
|
|
{
|
|
|
|
if (Dvar::Var("net_natFix").get<bool>()) return 0;
|
|
|
|
return Network::GetPort();
|
|
|
|
}
|
|
|
|
|
|
|
|
Node::Node()
|
|
|
|
{
|
|
|
|
if (ZoneBuilder::IsEnabled()) return;
|
|
|
|
Dvar::Register<bool>("net_natFix", false, 0, "Fix node registration for certain firewalls/routers");
|
|
|
|
|
2022-05-05 10:03:14 -04:00
|
|
|
Scheduler::Loop([]
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
|
|
|
Node::StoreNodes(false);
|
2022-06-04 08:59:14 -04:00
|
|
|
}, Scheduler::Pipeline::ASYNC);
|
2018-10-09 04:53:15 -04:00
|
|
|
|
2022-06-25 12:21:12 -04:00
|
|
|
Scheduler::Loop(Node::RunFrame, Scheduler::Pipeline::MAIN);
|
2022-05-05 10:03:14 -04:00
|
|
|
|
2018-10-09 04:53:15 -04:00
|
|
|
Session::Handle("nodeListResponse", Node::HandleResponse);
|
2022-06-25 12:21:12 -04:00
|
|
|
Session::Handle("nodeListRequest", [](const Network::Address& address, [[maybe_unused]] const std::string& data)
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
|
|
|
Node::SendList(address);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Load stored nodes
|
2022-06-16 10:15:26 -04:00
|
|
|
auto loadNodes = []
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
|
|
|
Node::LoadNodePreset();
|
|
|
|
Node::LoadNodes();
|
|
|
|
};
|
|
|
|
|
2022-10-15 16:31:16 -04:00
|
|
|
Scheduler::OnGameInitialized(loadNodes, Scheduler::Pipeline::MAIN);
|
2018-10-09 04:53:15 -04:00
|
|
|
|
2023-06-16 15:56:18 -04:00
|
|
|
Command::Add("listNodes", [](const Command::Params*)
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
2022-06-12 17:07:53 -04:00
|
|
|
Logger::Print("Nodes: {}\n", Node::Nodes.size());
|
2018-10-09 04:53:15 -04:00
|
|
|
|
2022-06-16 10:15:26 -04:00
|
|
|
std::lock_guard _(Node::Mutex);
|
2023-06-16 15:56:18 -04:00
|
|
|
for (const auto& node : Node::Nodes)
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
2022-08-19 19:10:35 -04:00
|
|
|
Logger::Print("{}\t({})\n", node.address.getString(), node.isValid() ? "Valid" : "Invalid");
|
2018-10-09 04:53:15 -04:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-06-16 15:56:18 -04:00
|
|
|
Command::Add("addNode", [](const Command::Params* params)
|
2018-10-09 04:53:15 -04:00
|
|
|
{
|
2022-03-17 14:50:20 -04:00
|
|
|
if (params->size() < 2) return;
|
2023-05-14 12:42:42 -04:00
|
|
|
auto address = Network::Address{ params->get(1) };
|
|
|
|
if (address.isValid())
|
|
|
|
{
|
|
|
|
Node::Add(address);
|
|
|
|
}
|
2018-10-09 04:53:15 -04:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
Node::~Node()
|
|
|
|
{
|
2022-06-16 10:15:26 -04:00
|
|
|
std::lock_guard _(Node::Mutex);
|
2018-10-09 04:53:15 -04:00
|
|
|
Node::StoreNodes(true);
|
|
|
|
Node::Nodes.clear();
|
|
|
|
}
|
|
|
|
}
|