stk-code_catmod/src/network/servers_manager.cpp

560 lines
22 KiB
C++
Raw Normal View History

//
// SuperTuxKart - a fun racing game with go-kart
2015-03-29 20:31:42 -04:00
// Copyright (C) 2013-2015 Glenn De Jonghe
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation; either version 3
// of the License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#include "network/servers_manager.hpp"
#include "config/stk_config.hpp"
#include "config/user_config.hpp"
#include "io/xml_node.hpp"
#include "network/network.hpp"
#include "network/network_config.hpp"
#include "network/network_string.hpp"
#include "network/server.hpp"
#include "network/socket_address.hpp"
2016-11-07 17:52:39 -05:00
#include "network/stk_host.hpp"
#include "network/stk_ipv6.hpp"
#include "online/xml_request.hpp"
#include "online/request_manager.hpp"
#include "utils/translation.hpp"
#include "utils/time.hpp"
2015-10-09 01:47:18 -04:00
#include <assert.h>
#include <functional>
#include <set>
2015-10-09 01:47:18 -04:00
#include <string>
#include <thread>
2015-10-09 01:47:18 -04:00
2018-04-25 14:26:23 -04:00
#if defined(WIN32)
# undef _WIN32_WINNT
# define _WIN32_WINNT 0x600
# include <iphlpapi.h>
#else
# include <ifaddrs.h>
# include <net/if.h>
2018-04-25 14:26:23 -04:00
#endif
2019-05-06 04:41:29 -04:00
const int64_t SERVER_REFRESH_INTERVAL = 5000;
static ServersManager* g_manager_singleton(NULL);
// ============================================================================
ServersManager* ServersManager::get()
{
if (g_manager_singleton == NULL)
g_manager_singleton = new ServersManager();
return g_manager_singleton;
} // get
// ============================================================================
void ServersManager::deallocate()
{
delete g_manager_singleton;
g_manager_singleton = NULL;
} // deallocate
// ----------------------------------------------------------------------------
ServersManager::ServersManager()
{
reset();
} // ServersManager
// ----------------------------------------------------------------------------
ServersManager::~ServersManager()
{
} // ~ServersManager
// ----------------------------------------------------------------------------
2015-11-09 04:51:00 -05:00
/** Returns a WAN update-list-of-servers request. It queries the
* STK server for an up-to-date list of servers.
*/
std::shared_ptr<Online::XMLRequest> ServersManager::getWANRefreshRequest() const
{
2015-11-09 04:51:00 -05:00
// ========================================================================
/** A small local class that triggers an update of the ServersManager
* when the request is finished. */
class WANRefreshRequest : public Online::XMLRequest
{
private:
// Run the ip detect in separate thread, so it can be done parallel
// with the wan server request (which takes few seconds too)
std::thread m_ip_detect_thread;
2015-11-09 04:51:00 -05:00
public:
WANRefreshRequest() : Online::XMLRequest(/*priority*/100)
{
m_ip_detect_thread = std::thread(std::bind(
&NetworkConfig::detectIPType, NetworkConfig::get()));
}
~WANRefreshRequest()
{
if (m_ip_detect_thread.joinable())
m_ip_detect_thread.join();
}
2015-11-09 04:51:00 -05:00
// --------------------------------------------------------------------
virtual void afterOperation() OVERRIDE
{
Online::XMLRequest::afterOperation();
if (m_ip_detect_thread.joinable())
m_ip_detect_thread.join();
ServersManager::get()->setWanServers(isSuccess(), getXMLData());
} // afterOperation
2015-11-09 04:51:00 -05:00
// --------------------------------------------------------------------
}; // RefreshRequest
2015-11-09 04:51:00 -05:00
// ========================================================================
auto request = std::make_shared<WANRefreshRequest>();
request->setApiURL(Online::API::SERVER_PATH, "get-all");
2015-11-09 04:51:00 -05:00
return request;
} // getWANRefreshRequest
// ----------------------------------------------------------------------------
2015-11-09 04:51:00 -05:00
/** Returns a LAN update-list-of-servers request. It uses UDP broadcasts
* to find LAN servers, and waits for a certain amount of time fr
* answers.
*/
std::shared_ptr<Online::XMLRequest> ServersManager::getLANRefreshRequest() const
2015-11-09 04:51:00 -05:00
{
/** A simple class that uses LAN broadcasts to find local servers.
* It is based on XML request, but actually does not use any of the
* XML/HTTP based infrastructure, but implements the same interface.
* This way the already existing request thread can be used.
*/
class LANRefreshRequest : public Online::XMLRequest
2015-11-09 04:51:00 -05:00
{
public:
/** High priority for this request. */
LANRefreshRequest() : XMLRequest(/*priority*/100) {m_success = false;}
2015-11-09 04:51:00 -05:00
// --------------------------------------------------------------------
virtual ~LANRefreshRequest() {}
// --------------------------------------------------------------------
/** Get the downloaded XML tree.
* \pre request has to be executed.
* \return get the complete result from the request reply. */
virtual const XMLNode * getXMLData() const
{
assert(hasBeenExecuted());
return NULL;
} // getXMLData
// --------------------------------------------------------------------
virtual void prepareOperation() OVERRIDE
{
} // prepareOperation
// --------------------------------------------------------------------
virtual void operation() OVERRIDE
2015-11-09 04:51:00 -05:00
{
ENetAddress addr = {};
setIPv6Socket(UserConfigParams::m_ipv6_lan ? 1 : 0);
NetworkConfig::get()->setIPType(UserConfigParams::m_ipv6_lan ?
NetworkConfig::IP_DUAL_STACK : NetworkConfig::IP_V4);
2016-11-07 17:52:39 -05:00
Network *broadcast = new Network(1, 1, 0, 0, &addr);
if (!broadcast->getENetHost())
{
setIPv6Socket(0);
m_success = true;
delete broadcast;
return;
}
const std::vector<SocketAddress> &all_bcast =
ServersManager::get()->getBroadcastAddresses(
UserConfigParams::m_ipv6_lan);
for (auto &bcast_addr : all_bcast)
{
Log::info("Server Discovery", "Broadcasting to %s",
bcast_addr.toString().c_str());
broadcast->sendRawPacket(std::string("stk-server"), bcast_addr);
}
2015-11-09 04:51:00 -05:00
Log::info("ServersManager", "Sent broadcast message.");
const int LEN=2048;
char buffer[LEN];
// Wait for up to 0.5 seconds to receive an answer from
// any local servers.
uint64_t start_time = StkTime::getMonoTimeMs();
const uint64_t DURATION = 1000;
const auto& servers = ServersManager::get()->getServers();
2018-03-16 07:06:20 -04:00
int cur_server_id = (int)servers.size();
2018-03-02 02:21:27 -05:00
assert(cur_server_id == 0);
// Use a map with the server name as key to automatically remove
// duplicated answers from a server (since we potentially do
// multiple broadcasts). We can not use the sender ip address,
// because e.g. a local client would answer as 127.0.0.1 and
// 192.168.**.
std::map<irr::core::stringw, std::shared_ptr<Server> > servers_now;
while (StkTime::getMonoTimeMs() - start_time < DURATION)
2015-11-09 04:51:00 -05:00
{
SocketAddress sender;
2015-11-09 04:51:00 -05:00
int len = broadcast->receiveRawPacket(buffer, LEN, &sender, 1);
if (len > 0)
2015-11-09 04:51:00 -05:00
{
2016-02-24 17:48:47 -05:00
BareNetworkString s(buffer, len);
2018-08-03 02:01:36 -04:00
int version = s.getUInt32();
if (version < stk_config->m_max_server_version ||
version > stk_config->m_max_server_version)
{
Log::verbose("ServersManager", "Skipping a server");
continue;
}
irr::core::stringw name;
2016-02-24 17:48:47 -05:00
// bytes_read is the number of bytes read
s.decodeStringW(&name);
uint8_t max_players = s.getUInt8();
uint8_t players = s.getUInt8();
uint16_t port = s.getUInt16();
uint8_t difficulty = s.getUInt8();
2018-03-02 02:21:27 -05:00
uint8_t mode = s.getUInt8();
sender.setPort(port);
uint8_t password = s.getUInt8();
uint8_t game_started = s.getUInt8();
2019-01-04 04:40:25 -05:00
std::string current_track;
try
{
s.decodeString(&current_track);
}
catch (std::exception& e)
{
(void)e;
}
auto server = std::make_shared<Server>(cur_server_id++,
name, max_players, players, difficulty, mode,
SocketAddress(sender.getIP(), sender.getPort()),
password == 1, game_started == 1, current_track);
if (sender.isIPv6())
{
server->setIPV6Address(sender);
server->setIPV6Connection(true);
}
servers_now.insert(std::make_pair(name, server));
//all_servers.[name] = servers_now.back();
2015-11-09 04:51:00 -05:00
} // if received_data
} // while still waiting
setIPv6Socket(0);
m_success = true;
ServersManager::get()->setLanServers(servers_now);
2016-06-27 19:47:56 -04:00
delete broadcast;
2015-11-09 04:51:00 -05:00
} // operation
// --------------------------------------------------------------------
/** This function is necessary, otherwise the XML- and HTTP-Request
* functions are called, which will cause a crash. */
virtual void afterOperation() OVERRIDE {}
// --------------------------------------------------------------------
}; // LANRefreshRequest
// ========================================================================
return std::make_shared<LANRefreshRequest>();
2015-11-09 04:51:00 -05:00
} // getLANRefreshRequest
// ----------------------------------------------------------------------------
/** Takes a mapping of server name to server data (to avoid having the same
* server listed more than once since the client will be doing multiple
* broadcasts to find a server), and converts this into a list of servers.
* \param servers Mapping of server name to Server object.
*/
void ServersManager::setLanServers(const std::map<irr::core::stringw,
std::shared_ptr<Server> >& servers)
{
m_servers.clear();
for (auto i : servers) m_servers.emplace_back(i.second);
m_last_load_time.store(StkTime::getMonoTimeMs());
m_list_updated = true;
}
// ----------------------------------------------------------------------------
2015-11-09 04:51:00 -05:00
/** Factory function to create either a LAN or a WAN update-of-server
* requests. The current list of servers is also cleared.
2015-11-09 04:51:00 -05:00
*/
2018-04-25 14:26:23 -04:00
bool ServersManager::refresh(bool full_refresh)
2015-11-09 04:51:00 -05:00
{
2019-05-06 04:41:29 -04:00
if ((int64_t)StkTime::getMonoTimeMs() - m_last_load_time.load()
2015-11-09 04:51:00 -05:00
< SERVER_REFRESH_INTERVAL)
{
// Avoid too frequent refreshing
return false;
}
2015-11-09 04:51:00 -05:00
cleanUpServers();
m_list_updated = false;
2019-05-06 04:41:29 -04:00
if (NetworkConfig::get()->isWAN())
2018-04-25 14:26:23 -04:00
{
Online::RequestManager::get()->addRequest(getWANRefreshRequest());
}
else
2018-04-25 14:26:23 -04:00
{
Online::RequestManager::get()->addRequest(getLANRefreshRequest());
}
return true;
} // refresh
// ----------------------------------------------------------------------------
/** Callback from the refresh request for wan servers.
* \param success If the refresh was successful.
* \param input The XML data describing the server.
*/
void ServersManager::setWanServers(bool success, const XMLNode* input)
{
if (!success)
{
Log::error("Server Manager", "Could not refresh server list");
m_list_updated = true;
return;
}
const XMLNode *servers_xml = input->getNode("servers");
for (unsigned int i = 0; i < servers_xml->getNumNodes(); i++)
{
2018-09-03 19:37:21 -04:00
const XMLNode* s = servers_xml->getNode(i);
assert(s);
const XMLNode* si = s->getNode("server-info");
assert(si);
int version = 0;
2018-09-03 19:37:21 -04:00
si->get("version", &version);
assert(version != 0);
if (version < stk_config->m_max_server_version ||
version > stk_config->m_max_server_version)
{
Log::verbose("ServersManager", "Skipping a server");
continue;
}
std::shared_ptr<Server> ser = std::make_shared<Server>(*s);
if (ser->getAddress().isUnset() &&
NetworkConfig::get()->getIPType() == NetworkConfig::IP_V4)
{
Log::verbose("ServersManager", "Skipping an IPv6 only server");
continue;
}
m_servers.emplace_back(ser);
}
m_last_load_time.store(StkTime::getMonoTimeMs());
m_list_updated = true;
} // refresh
2018-04-25 14:26:23 -04:00
// ----------------------------------------------------------------------------
/** Sets a list of default broadcast addresses which is used in case no valid
* broadcast address is found. This list includes default private network
* addresses.
*/
std::vector<SocketAddress> ServersManager::getDefaultBroadcastAddresses()
2018-04-25 14:26:23 -04:00
{
// Add some common LAN addresses
std::vector<SocketAddress> result;
uint16_t port = stk_config->m_server_discovery_port;
result.emplace_back(std::string("192.168.255.255"), port);
result.emplace_back(std::string("192.168.0.255"), port);
result.emplace_back(std::string("192.168.1.255"), port);
result.emplace_back(std::string("172.31.255.255"), port);
result.emplace_back(std::string("172.16.255.255"), port);
result.emplace_back(std::string("172.16.0.255"), port);
result.emplace_back(std::string("10.255.255.255"), port);
result.emplace_back(std::string("10.0.255.255"), port);
result.emplace_back(std::string("10.0.0.255"), port);
result.emplace_back(std::string("255.255.255.255"), port);
result.emplace_back(std::string("127.0.0.255"), port);
result.emplace_back(std::string("127.0.0.1"), port);
return result;
} // getDefaultBroadcastAddresses
2018-04-25 14:26:23 -04:00
// ----------------------------------------------------------------------------
/** This masks various possible broadcast addresses. For example, in a /16
* network it would first use *.*.255.255, then *.*.*.255. Also if the
* length of the mask is not a multiple of 8, the original value will
* be used, before multiple of 8 are create: /22 (*.3f.ff.ff), then
* /16 (*.*.ff.ff), /8 (*.*.*.ff). While this is usually an overkill,
* it can help in the case that the router does not forward a broadcast
* as expected (this problem often happens with 255.255.255.255, which is
* why this broadcast address creation code was added).
* \param a The transport address for which the broadcast addresses need
* to be created.
* \param len Number of bits to be or'ed.
* \param result Location to put address.
2018-04-25 14:26:23 -04:00
*/
void ServersManager::addAllBroadcastAddresses(const SocketAddress &a, int len,
std::vector<SocketAddress>* result)
2018-04-25 14:26:23 -04:00
{
// Try different broadcast addresses - by masking on
// byte boundaries
while (len > 0)
{
unsigned int mask = (1 << len) - 1;
SocketAddress bcast(a.getIP() | mask,
2018-09-11 02:06:30 -04:00
stk_config->m_server_discovery_port);
2018-04-25 14:26:23 -04:00
Log::info("Broadcast", "address %s length %d mask %x --> %s",
a.toString().c_str(),
len, mask,
bcast.toString().c_str());
result->push_back(bcast);
2018-04-25 14:26:23 -04:00
if (len % 8 != 0)
len -= (len % 8);
else
len = len - 8;
} // while len > 0
} // addAllBroadcastAddresses
// ----------------------------------------------------------------------------
/** Returns a list of all possible broadcast addresses on this machine.
* It queries all adapters for active IPv4 interfaces, determines their
2018-04-25 14:26:23 -04:00
* netmask to create the broadcast addresses. It will also add 'smaller'
* broadcast addesses, e.g. in a /16 network, it will add *.*.255.255 and
* *.*.*.255, since it was sometimes observed that routers would not let
* all broadcast addresses through. Duplicated answers (from the same server
* to different addersses) will be filtered out in ServersManager.
*/
std::vector<SocketAddress> ServersManager::getBroadcastAddresses(bool ipv6)
2018-04-25 14:26:23 -04:00
{
std::vector<SocketAddress> result;
#ifndef WIN32
2018-04-25 14:26:23 -04:00
struct ifaddrs *addresses, *p;
2019-01-13 13:26:30 -05:00
if (getifaddrs(&addresses) == -1)
{
Log::warn("SocketAddress", "Error in getifaddrs");
return result;
2019-01-13 13:26:30 -05:00
}
std::set<uint32_t> used_scope_id;
2018-04-25 14:26:23 -04:00
for (p = addresses; p; p = p->ifa_next)
{
SocketAddress socket_address;
if (p->ifa_addr == NULL)
continue;
if (p->ifa_addr->sa_family == AF_INET)
2018-04-25 14:26:23 -04:00
{
struct sockaddr_in *sa = (struct sockaddr_in *) p->ifa_addr;
uint32_t addr = htonl(sa->sin_addr.s_addr);
// Skip 169.254.*.* local link address
if (((addr >> 24) & 0xff) == 169 &&
((addr >> 16) & 0xff) == 254)
continue;
SocketAddress saddr(addr, 0);
2018-04-25 14:26:23 -04:00
uint32_t u = ((sockaddr_in*)(p->ifa_netmask))->sin_addr.s_addr;
// Convert mask to #bits: SWAT algorithm
u = u - ((u >> 1) & 0x55555555);
u = (u & 0x33333333) + ((u >> 2) & 0x33333333);
u = (((u + (u >> 4)) & 0x0F0F0F0F) * 0x01010101) >> 24;
Log::debug("ServerManager",
"Interface: %s\tAddress: %s\tmask: %x\n", p->ifa_name,
saddr.toString().c_str(), u);
addAllBroadcastAddresses(saddr, u, &result);
}
else if (p->ifa_addr->sa_family == AF_INET6 && ipv6)
{
uint32_t idx = if_nametoindex(p->ifa_name);
if (used_scope_id.find(idx) != used_scope_id.end())
continue;
used_scope_id.insert(idx);
SocketAddress socket_address("ff02::1",
stk_config->m_server_discovery_port);
sockaddr_in6* in6 = (sockaddr_in6*)socket_address.getSockaddr();
in6->sin6_scope_id = idx;
result.push_back(socket_address);
2018-04-25 14:26:23 -04:00
}
}
2018-07-11 01:49:01 -04:00
freeifaddrs(addresses);
#else
// From docs from microsoft it recommends 15k size
const int WORKING_BUFFER_SIZE = 15000;
PIP_ADAPTER_ADDRESSES paddr = NULL;
unsigned long len = WORKING_BUFFER_SIZE;
int return_code = 0;
int iteration = 0;
do
{
paddr = (IP_ADAPTER_ADDRESSES*)malloc(len);
if (paddr == NULL)
return result;
long flags = 0;
return_code = GetAdaptersAddresses(AF_UNSPEC, flags, NULL, paddr,
&len);
if (return_code == ERROR_BUFFER_OVERFLOW)
{
free(paddr);
paddr = NULL;
}
else
break;
iteration++;
} while ((return_code == ERROR_BUFFER_OVERFLOW) && (iteration < 10));
if (return_code == ERROR_BUFFER_OVERFLOW)
{
Log::warn("ServerManager", "Can not get broadcast addresses.");
result = getDefaultBroadcastAddresses();
paddr = NULL;
}
for (IP_ADAPTER_ADDRESSES *p = paddr; p; p = p->Next)
{
if (p->OperStatus != IfOperStatusUp)
continue;
std::set<uint32_t> used_scope_id;
for (PIP_ADAPTER_UNICAST_ADDRESS unicast = p->FirstUnicastAddress;
unicast != NULL; unicast = unicast->Next)
{
SocketAddress socket_address;
if (unicast->Address.lpSockaddr->sa_family == AF_INET)
{
const sockaddr_in *sa =
(const sockaddr_in*)unicast->Address.lpSockaddr;
// Skip 169.254.*.* local link address
if (sa->sin_addr.S_un.S_un_b.s_b1 == 169 &&
sa->sin_addr.S_un.S_un_b.s_b2 == 254)
continue;
// Use sa->sin_addr.S_un.S_addr and htonl?
SocketAddress ta(sa->sin_addr.S_un.S_un_b.s_b1,
sa->sin_addr.S_un.S_un_b.s_b2,
sa->sin_addr.S_un.S_un_b.s_b3,
sa->sin_addr.S_un.S_un_b.s_b4);
int len = 32 - unicast->OnLinkPrefixLength;
addAllBroadcastAddresses(ta, len, &result);
}
if (unicast->Address.lpSockaddr->sa_family == AF_INET6 && ipv6)
{
sockaddr_in6* in6 =
(sockaddr_in6*)unicast->Address.lpSockaddr;
const uint32_t scope_id = in6->sin6_scope_id;
if (used_scope_id.find(scope_id) !=
used_scope_id.end())
continue;
used_scope_id.insert(scope_id);
SocketAddress socket_address("ff02::1",
stk_config->m_server_discovery_port);
in6 = (sockaddr_in6*)socket_address.getSockaddr();
in6->sin6_scope_id = scope_id;
result.push_back(socket_address);
}
}
}
free(paddr);
2018-04-25 14:26:23 -04:00
#endif
if (ipv6)
{
// Convert IPv4 socket address to ::ffff:x.y.z.w
for (auto& addr : result)
addr.convertForIPv6Socket(ipv6);
2018-04-25 14:26:23 -04:00
}
return result;
2018-04-25 14:26:23 -04:00
} // getBroadcastAddresses