mirror of
https://github.com/vcmi/vcmi.git
synced 2026-04-26 20:02:20 +02:00
210 lines
5.4 KiB
C++
210 lines
5.4 KiB
C++
/*
|
|
* NetworkDiscovery.cpp, part of VCMI engine
|
|
*
|
|
* Authors: listed in file AUTHORS in main folder
|
|
*
|
|
* License: GNU General Public License v2.0 or later
|
|
* Full text of license available in license.txt file, in main folder
|
|
*
|
|
*/
|
|
|
|
#include "StdInc.h"
|
|
#include "NetworkDiscovery.h"
|
|
#include "NetworkInterface.h"
|
|
|
|
VCMI_LIB_NAMESPACE_BEGIN
|
|
|
|
ServerDiscovery::ServerDiscovery(NetworkContext & context, IServerDiscoveryObserver & observer)
|
|
: context(context), observer(observer)
|
|
{
|
|
}
|
|
|
|
ServerDiscovery::~ServerDiscovery()
|
|
{
|
|
abort();
|
|
}
|
|
|
|
void ServerDiscovery::abort()
|
|
{
|
|
if(sendTimer)
|
|
{
|
|
#if BOOST_VERSION >= 108700
|
|
sendTimer->cancel();
|
|
#else
|
|
boost::system::error_code ec;
|
|
sendTimer->cancel(ec);
|
|
#endif
|
|
}
|
|
if(socket)
|
|
{
|
|
boost::system::error_code ec;
|
|
socket->close(ec);
|
|
socket.reset();
|
|
}
|
|
}
|
|
|
|
void ServerDiscovery::start()
|
|
{
|
|
auto self = shared_from_this();
|
|
socket = std::make_shared<boost::asio::ip::udp::socket>(context);
|
|
socket->open(boost::asio::ip::udp::v4());
|
|
socket->set_option(boost::asio::socket_base::broadcast(true));
|
|
|
|
auto recvBuf = std::make_shared<std::array<char, 1024>>();
|
|
auto senderEndpoint = std::make_shared<boost::asio::ip::udp::endpoint>();
|
|
sendTimer = std::make_shared<boost::asio::steady_timer>(context);
|
|
|
|
// Receive handler for incoming server responses
|
|
auto receiveHandler = std::make_shared<std::function<void(const boost::system::error_code&, std::size_t)>>();
|
|
*receiveHandler = [self, socket = this->socket, recvBuf, senderEndpoint, receiveHandler](const boost::system::error_code& ec, std::size_t len)
|
|
{
|
|
if(ec == boost::asio::error::operation_aborted)
|
|
return;
|
|
|
|
if(!ec && len > 0)
|
|
{
|
|
std::string resp(recvBuf->data(), recvBuf->data() + len);
|
|
if(resp.rfind("VCMI_SERVER:", 0) == 0)
|
|
{
|
|
std::string portStr = resp.substr(12);
|
|
DiscoveredServer server;
|
|
server.address = senderEndpoint->address().to_string();
|
|
server.port = static_cast<uint16_t>(std::stoi(portStr));
|
|
|
|
auto serverKey = std::make_pair(server.address, server.port);
|
|
if(self->discoveredServers.insert(serverKey).second)
|
|
{
|
|
logGlobal->info("Discovered server: %s", resp.c_str());
|
|
self->observer.onServerDiscovered(server);
|
|
}
|
|
}
|
|
}
|
|
socket->async_receive_from(boost::asio::buffer(*recvBuf), *senderEndpoint, *receiveHandler);
|
|
};
|
|
|
|
// Send handler for broadcasting discovery message
|
|
auto sendHandler = std::make_shared<std::function<void(const boost::system::error_code&)>>();
|
|
*sendHandler = [self, socket = this->socket, sendHandler](const boost::system::error_code& ec)
|
|
{
|
|
if(ec == boost::asio::error::operation_aborted || ec == boost::asio::error::bad_descriptor)
|
|
return;
|
|
|
|
if(ec)
|
|
{
|
|
logGlobal->error("Discovery send error: %s", ec.message());
|
|
return;
|
|
}
|
|
|
|
std::string message = "VCMI_DISCOVERY";
|
|
boost::asio::ip::udp::endpoint broadcastEndpoint(boost::asio::ip::address_v4::broadcast(), 3030);
|
|
socket->async_send_to(boost::asio::buffer(message), broadcastEndpoint,
|
|
[self, sendHandler](const boost::system::error_code& ec, std::size_t)
|
|
{
|
|
if(ec && ec != boost::asio::error::operation_aborted && ec != boost::asio::error::bad_descriptor)
|
|
logGlobal->error("Discovery send error: %s", ec.message());
|
|
else if(!ec)
|
|
{
|
|
// Schedule next broadcast in 1 second
|
|
self->sendTimer->expires_after(std::chrono::seconds(1));
|
|
self->sendTimer->async_wait(*sendHandler);
|
|
}
|
|
}
|
|
);
|
|
};
|
|
|
|
// Start receiving
|
|
socket->async_receive_from(boost::asio::buffer(*recvBuf), *senderEndpoint, *receiveHandler);
|
|
|
|
// Send first broadcast immediately
|
|
(*sendHandler)(boost::system::error_code());
|
|
}
|
|
|
|
std::vector<std::string> ServerDiscovery::ipAddresses()
|
|
{
|
|
std::vector<std::string> addresses;
|
|
|
|
try
|
|
{
|
|
boost::asio::io_context io;
|
|
boost::asio::ip::udp::socket socket(io);
|
|
|
|
socket.connect({
|
|
boost::asio::ip::make_address("8.8.8.8"), 53
|
|
});
|
|
|
|
auto addr = socket.local_endpoint().address();
|
|
if (addr.is_v4())
|
|
addresses.push_back(addr.to_string());
|
|
}
|
|
catch (const std::exception& e)
|
|
{
|
|
logGlobal->error("IP address retrieval error: %s", e.what());
|
|
}
|
|
|
|
return addresses;
|
|
}
|
|
|
|
ServerDiscoveryListener::ServerDiscoveryListener(NetworkContext & context, IServerDiscoveryAnnouncer & announcer, uint16_t port)
|
|
: context(context), announcer(announcer), port(port)
|
|
{
|
|
}
|
|
|
|
ServerDiscoveryListener::~ServerDiscoveryListener()
|
|
{
|
|
stop();
|
|
}
|
|
|
|
void ServerDiscoveryListener::start()
|
|
{
|
|
if(socket)
|
|
return;
|
|
|
|
boost::asio::ip::udp::endpoint listenEndpoint(boost::asio::ip::udp::v4(), port);
|
|
|
|
socket = std::make_shared<boost::asio::ip::udp::socket>(context, listenEndpoint);
|
|
asyncReceive();
|
|
}
|
|
|
|
void ServerDiscoveryListener::stop()
|
|
{
|
|
if(socket)
|
|
{
|
|
boost::system::error_code ec;
|
|
socket->close(ec);
|
|
socket.reset();
|
|
}
|
|
}
|
|
|
|
void ServerDiscoveryListener::asyncReceive()
|
|
{
|
|
auto self = shared_from_this();
|
|
auto socketPtr = socket;
|
|
if(!socketPtr)
|
|
return;
|
|
|
|
socketPtr->async_receive_from(
|
|
boost::asio::buffer(recvBuffer),
|
|
remoteEndpoint,
|
|
[self, socketPtr, this](const boost::system::error_code & ec, std::size_t len)
|
|
{
|
|
if(ec == boost::asio::error::operation_aborted)
|
|
return;
|
|
|
|
if(!ec && len > 0)
|
|
{
|
|
std::string msg(recvBuffer.data(), recvBuffer.data() + len);
|
|
|
|
if(msg == "VCMI_DISCOVERY" && self->announcer.isInLobby())
|
|
{
|
|
std::string response = "VCMI_SERVER:" + std::to_string(self->announcer.getPort());
|
|
socketPtr->send_to(boost::asio::buffer(response), remoteEndpoint);
|
|
}
|
|
}
|
|
|
|
self->asyncReceive();
|
|
}
|
|
);
|
|
}
|
|
|
|
VCMI_LIB_NAMESPACE_END
|