NetworkDiscovery.cpp 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. /*
  2. * NetworkDiscovery.cpp, part of VCMI engine
  3. *
  4. * Authors: listed in file AUTHORS in main folder
  5. *
  6. * License: GNU General Public License v2.0 or later
  7. * Full text of license available in license.txt file, in main folder
  8. *
  9. */
  10. #include "StdInc.h"
  11. #include "NetworkDiscovery.h"
  12. #include "NetworkInterface.h"
  13. VCMI_LIB_NAMESPACE_BEGIN
  14. ServerDiscovery::ServerDiscovery(NetworkContext & context, IServerDiscoveryObserver & observer)
  15. : context(context), observer(observer)
  16. {
  17. }
  18. ServerDiscovery::~ServerDiscovery()
  19. {
  20. abort();
  21. }
  22. void ServerDiscovery::abort()
  23. {
  24. if(sendTimer)
  25. {
  26. #if BOOST_VERSION >= 108700
  27. sendTimer->cancel();
  28. #else
  29. boost::system::error_code ec;
  30. sendTimer->cancel(ec);
  31. #endif
  32. }
  33. if(socket)
  34. {
  35. boost::system::error_code ec;
  36. socket->close(ec);
  37. socket.reset();
  38. }
  39. }
  40. void ServerDiscovery::start()
  41. {
  42. auto self = shared_from_this();
  43. socket = std::make_shared<boost::asio::ip::udp::socket>(context);
  44. socket->open(boost::asio::ip::udp::v4());
  45. socket->set_option(boost::asio::socket_base::broadcast(true));
  46. auto recvBuf = std::make_shared<std::array<char, 1024>>();
  47. auto senderEndpoint = std::make_shared<boost::asio::ip::udp::endpoint>();
  48. sendTimer = std::make_shared<boost::asio::steady_timer>(context);
  49. // Receive handler for incoming server responses
  50. auto receiveHandler = std::make_shared<std::function<void(const boost::system::error_code&, std::size_t)>>();
  51. *receiveHandler = [self, socket = this->socket, recvBuf, senderEndpoint, receiveHandler](const boost::system::error_code& ec, std::size_t len)
  52. {
  53. if(ec == boost::asio::error::operation_aborted)
  54. return;
  55. if(!ec && len > 0)
  56. {
  57. std::string resp(recvBuf->data(), recvBuf->data() + len);
  58. if(resp.rfind("VCMI_SERVER:", 0) == 0)
  59. {
  60. std::string portStr = resp.substr(12);
  61. DiscoveredServer server;
  62. server.address = senderEndpoint->address().to_string();
  63. server.port = static_cast<uint16_t>(std::stoi(portStr));
  64. auto serverKey = std::make_pair(server.address, server.port);
  65. if(self->discoveredServers.insert(serverKey).second)
  66. {
  67. logGlobal->info("Discovered server: %s", resp.c_str());
  68. self->observer.onServerDiscovered(server);
  69. }
  70. }
  71. }
  72. socket->async_receive_from(boost::asio::buffer(*recvBuf), *senderEndpoint, *receiveHandler);
  73. };
  74. // Send handler for broadcasting discovery message
  75. auto sendHandler = std::make_shared<std::function<void(const boost::system::error_code&)>>();
  76. *sendHandler = [self, socket = this->socket, sendHandler](const boost::system::error_code& ec)
  77. {
  78. if(ec == boost::asio::error::operation_aborted || ec == boost::asio::error::bad_descriptor)
  79. return;
  80. if(ec)
  81. {
  82. logGlobal->error("Discovery send error: %s", ec.message());
  83. return;
  84. }
  85. std::string message = "VCMI_DISCOVERY";
  86. boost::asio::ip::udp::endpoint broadcastEndpoint(boost::asio::ip::address_v4::broadcast(), 3030);
  87. socket->async_send_to(boost::asio::buffer(message), broadcastEndpoint,
  88. [self, sendHandler](const boost::system::error_code& ec, std::size_t)
  89. {
  90. if(ec && ec != boost::asio::error::operation_aborted && ec != boost::asio::error::bad_descriptor)
  91. logGlobal->error("Discovery send error: %s", ec.message());
  92. else if(!ec)
  93. {
  94. // Schedule next broadcast in 1 second
  95. self->sendTimer->expires_after(std::chrono::seconds(1));
  96. self->sendTimer->async_wait(*sendHandler);
  97. }
  98. }
  99. );
  100. };
  101. // Start receiving
  102. socket->async_receive_from(boost::asio::buffer(*recvBuf), *senderEndpoint, *receiveHandler);
  103. // Send first broadcast immediately
  104. (*sendHandler)(boost::system::error_code());
  105. }
  106. std::vector<std::string> ServerDiscovery::ipAddresses()
  107. {
  108. std::vector<std::string> addresses;
  109. try
  110. {
  111. boost::asio::io_context io;
  112. boost::asio::ip::udp::socket socket(io);
  113. socket.connect({
  114. boost::asio::ip::make_address("8.8.8.8"), 53
  115. });
  116. auto addr = socket.local_endpoint().address();
  117. if (addr.is_v4())
  118. addresses.push_back(addr.to_string());
  119. }
  120. catch (const std::exception& e)
  121. {
  122. logGlobal->error("IP address retrieval error: %s", e.what());
  123. }
  124. return addresses;
  125. }
  126. ServerDiscoveryListener::ServerDiscoveryListener(NetworkContext & context, IServerDiscoveryAnnouncer & announcer, uint16_t port)
  127. : context(context), announcer(announcer), port(port)
  128. {
  129. }
  130. ServerDiscoveryListener::~ServerDiscoveryListener()
  131. {
  132. stop();
  133. }
  134. void ServerDiscoveryListener::start()
  135. {
  136. if(socket)
  137. return;
  138. boost::asio::ip::udp::endpoint listenEndpoint(boost::asio::ip::udp::v4(), port);
  139. socket = std::make_shared<boost::asio::ip::udp::socket>(context, listenEndpoint);
  140. asyncReceive();
  141. }
  142. void ServerDiscoveryListener::stop()
  143. {
  144. if(socket)
  145. {
  146. boost::system::error_code ec;
  147. socket->close(ec);
  148. socket.reset();
  149. }
  150. }
  151. void ServerDiscoveryListener::asyncReceive()
  152. {
  153. auto self = shared_from_this();
  154. auto socketPtr = socket;
  155. if(!socketPtr)
  156. return;
  157. socketPtr->async_receive_from(
  158. boost::asio::buffer(recvBuffer),
  159. remoteEndpoint,
  160. [self, socketPtr, this](const boost::system::error_code & ec, std::size_t len)
  161. {
  162. if(ec == boost::asio::error::operation_aborted)
  163. return;
  164. if(!ec && len > 0)
  165. {
  166. std::string msg(recvBuffer.data(), recvBuffer.data() + len);
  167. if(msg == "VCMI_DISCOVERY" && self->announcer.isInLobby())
  168. {
  169. std::string response = "VCMI_SERVER:" + std::to_string(self->announcer.getPort());
  170. socketPtr->send_to(boost::asio::buffer(response), remoteEndpoint);
  171. }
  172. }
  173. self->asyncReceive();
  174. }
  175. );
  176. }
  177. VCMI_LIB_NAMESPACE_END