#include "STDInclude.hpp" namespace Components { Discovery::Container Discovery::DiscoveryContainer = { false, false, std::thread() }; void Discovery::Perform() { Discovery::DiscoveryContainer.Perform = true; } Discovery::Discovery() { Dvar::Register("net_discoveryPortRangeMin", 25000, 0, 65535, Game::dvar_flag::DVAR_FLAG_SAVED, "Minimum scan range port for local server discovery"); Dvar::Register("net_discoveryPortRangeMax", 35000, 1, 65536, Game::dvar_flag::DVAR_FLAG_SAVED, "Maximum scan range port for local server discovery"); // An additional thread prevents lags // Not sure if that's the best way though Discovery::DiscoveryContainer.Perform = false; Discovery::DiscoveryContainer.Terminate = false; Discovery::DiscoveryContainer.Thread = std::thread([] () { while (!Discovery::DiscoveryContainer.Terminate) { if (Discovery::DiscoveryContainer.Perform) { int start = Game::Sys_Milliseconds(); Logger::Print("Starting local server discovery...\n"); Discovery::DiscoveryContainer.Challenge = fmt::sprintf("%X", Utils::Cryptography::Rand::GenerateInt()); unsigned int minPort = Dvar::Var("net_discoveryPortRangeMin").Get(); unsigned int maxPort = Dvar::Var("net_discoveryPortRangeMax").Get(); Network::BroadcastRange(minPort, maxPort, fmt::sprintf("discovery %s", Discovery::DiscoveryContainer.Challenge.data())); Logger::Print("Discovery sent within %dms, awaiting responses...\n", Game::Sys_Milliseconds() - start); Discovery::DiscoveryContainer.Perform = false; } std::this_thread::sleep_for(50ms); } }); Network::Handle("discovery", [] (Network::Address address, std::string data) { if (address.IsSelf()) return; if (!address.IsLocal()) { Logger::Print("Received discovery request from non-local address: %s\n", address.GetCString()); return; } Logger::Print("Received discovery request from %s\n", address.GetCString()); Network::SendCommand(address, "discoveryResponse", data); }); Network::Handle("discoveryResponse", [] (Network::Address address, std::string data) { if (address.IsSelf()) return; if (!address.IsLocal()) { Logger::Print("Received discovery response from non-local address: %s\n", address.GetCString()); return; } if (Utils::ParseChallenge(data) != Discovery::DiscoveryContainer.Challenge) { Logger::Print("Received discovery with invalid challenge from: %s\n", address.GetCString()); return; } Logger::Print("Received discovery response from: %s\n", address.GetCString()); if (ServerList::IsOfflineList()) { ServerList::InsertRequest(address, true); } }); } Discovery::~Discovery() { Discovery::DiscoveryContainer.Perform = false; Discovery::DiscoveryContainer.Terminate = true; if (Discovery::DiscoveryContainer.Thread.joinable()) { Discovery::DiscoveryContainer.Thread.join(); } } }