// // Created by koncord on 21.04.17. // #include #include #include #include #include "MasterServer.hpp" #include #include #include #include #include #include using namespace RakNet; using namespace std; using namespace mwmp; using namespace chrono; MasterServer::MasterServer(const std::string &luaScript) { state.open_libraries(); boost::filesystem::path absPath = boost::filesystem::absolute(luaScript); std::string package_path = state["package"]["path"]; state["package"]["path"] = Utils::convertPath(absPath.parent_path().string() + "/?.lua") + ";" + package_path; state.set_function("BanAddress", &MasterServer::ban, this); state.set_function("UnbanAddress", &MasterServer::unban, this); state.script_file(luaScript); sol::table config = state["config"]; if (config.get_type() != sol::type::table) throw runtime_error("config is not correct"); sol::object maxConnections = config["maxConnections"]; if (maxConnections.get_type() != sol::type::number) throw runtime_error("config.maxConnections is not correct"); sol::object port = config["port"]; if (port.get_type() != sol::type::number) throw runtime_error("config.port is not correct"); state.new_usertype("Server", "name", sol::property(&MasterServer::SServer::GetName), "gamemode", sol::property(&MasterServer::SServer::GetGameMode), "version", sol::property(&MasterServer::SServer::GetVersion) ); peer = RakPeerInterface::GetInstance(); sockdescr = SocketDescriptor(port.as(), nullptr); peer->Startup(maxConnections.as(), &sockdescr, 1, 1000); peer->SetMaximumIncomingConnections(maxConnections.as()); peer->SetIncomingPassword(TES3MP_MASTERSERVER_PASSW, (int) strlen(TES3MP_MASTERSERVER_PASSW)); run = false; } MasterServer::~MasterServer() { Stop(true); } using namespace chrono; void MasterServer::Thread() { unsigned char packetId = 0; auto startTime = chrono::steady_clock::now(); BitStream send; PacketMasterQuery pmq(peer); pmq.SetSendStream(&send); PacketMasterUpdate pmu(peer); pmu.SetSendStream(&send); PacketMasterAnnounce pma(peer); pma.SetSendStream(&send); while (run) { Packet *packet = peer->Receive(); auto now = steady_clock::now(); if (now - startTime >= 60s) { startTime = steady_clock::now(); for (auto it = servers.begin(); it != servers.end();) { if (it->second.lastUpdate + 60s <= now) servers.erase(it++); else ++it; } for (auto id = pendingACKs.begin(); id != pendingACKs.end();) { if (now - id->second >= 30s) { cout << "timeout: " << peer->GetSystemAddressFromGuid(id->first).ToString() << endl; peer->CloseConnection(id->first, true); id = pendingACKs.erase(id); } else ++id; } } if (packet == nullptr) RakSleep(10); else for (; packet; peer->DeallocatePacket(packet), packet = peer->Receive()) { BitStream data(packet->data, packet->length, false); data.Read(packetId); switch (packetId) { case ID_NEW_INCOMING_CONNECTION: cout << "New incoming connection: " << packet->systemAddress.ToString() << endl; break; case ID_DISCONNECTION_NOTIFICATION: cout << "Disconnected: " << packet->systemAddress.ToString() << endl; break; case ID_CONNECTION_LOST: cout << "Connection lost: " << packet->systemAddress.ToString() << endl; break; case ID_MASTER_QUERY: { pmq.SetServers(reinterpret_cast *>(&servers)); pmq.Send(packet->systemAddress); pendingACKs[packet->guid] = steady_clock::now(); cout << "Sent info about all " << servers.size() << " servers to " << packet->systemAddress.ToString() << endl; break; } case ID_MASTER_UPDATE: { SystemAddress addr; data.Read(addr); // update 1 server ServerIter it = servers.find(addr); if (it != servers.end()) { pair pairPtr(it->first, static_cast(it->second)); pmu.SetServer(&pairPtr); pmu.Send(packet->systemAddress); pendingACKs[packet->guid] = steady_clock::now(); cout << "Sent info about " << addr.ToString() << " to " << packet->systemAddress.ToString() << endl; } break; } case ID_MASTER_ANNOUNCE: { ServerIter iter = servers.find(packet->systemAddress); pma.SetReadStream(&data); SServer server; pma.SetServer(&server); pma.Read(); auto keepAliveFunc = [&]() { iter->second.lastUpdate = now; pma.SetFunc(PacketMasterAnnounce::FUNCTION_KEEP); pma.Send(packet->systemAddress); pendingACKs[packet->guid] = steady_clock::now(); }; auto isServerValid = [&](const SServer &sserver) { bool ret = false; auto addr = packet->systemAddress.ToString(false); if (find(banned.begin(), banned.end(), addr) != banned.end()) // check if address is banned return false; luaStuff([&ret, &packet, &sserver, &addr](sol::state &state) { sol::protected_function func = state["OnServerAnnounce"]; sol::protected_function_result result = func.call(addr, sserver); if (result.valid()) ret = result.get(); else cerr << "Error: " << result.get() << endl; }); return ret; }; if (iter != servers.end()) { if (pma.GetFunc() == PacketMasterAnnounce::FUNCTION_DELETE) { servers.erase(iter); cout << "Deleted"; pma.Send(packet->systemAddress); pendingACKs[packet->guid] = steady_clock::now(); } else if (pma.GetFunc() == PacketMasterAnnounce::FUNCTION_ANNOUNCE) { cout << "Updated"; if (isServerValid(server)) { iter->second = server; keepAliveFunc(); } else { servers.erase(iter); pendingACKs[packet->guid] = steady_clock::now(); } } else { cout << "Keeping alive"; keepAliveFunc(); } } else if (pma.GetFunc() == PacketMasterAnnounce::FUNCTION_ANNOUNCE) { cout << "Added"; if (isServerValid(server)) { iter = servers.insert({packet->systemAddress, server}).first; keepAliveFunc(); } } else { cout << "Unknown"; pma.SetFunc(PacketMasterAnnounce::FUNCTION_DELETE); pma.Send(packet->systemAddress); pendingACKs[packet->guid] = steady_clock::now(); } cout << " server " << packet->systemAddress.ToString() << endl; break; } case ID_SND_RECEIPT_ACKED: uint32_t num; memcpy(&num, packet->data+1, 4); cout << "Packet with id " << num << " was delivered." << endl; pendingACKs.erase(packet->guid); peer->CloseConnection(packet->systemAddress, true); break; default: cout << "Wrong packet. id " << (unsigned) packet->data[0] << " packet length " << packet->length << " from " << packet->systemAddress.ToString() << endl; peer->CloseConnection(packet->systemAddress, true); } } } peer->Shutdown(1000); RakPeerInterface::DestroyInstance(peer); cout << "Server thread stopped" << endl; } void MasterServer::Start() { if (!run) { run = true; tMasterThread = thread(&MasterServer::Thread, this); cout << "Started" << endl; } } void MasterServer::Stop(bool wait) { if (run) { run = false; if (wait && tMasterThread.joinable()) tMasterThread.join(); } } bool MasterServer::isRunning() { return run; } void MasterServer::Wait() { if (run) { if (tMasterThread.joinable()) tMasterThread.join(); } } MasterServer::ServerMap *MasterServer::GetServers() { return &servers; } void MasterServer::luaStuff(std::function f) { lock_guard lock(luaMutex); f(state); } void MasterServer::ban(const std::string &addr) { banned.push_back(addr); } void MasterServer::unban(const std::string &addr) { auto it = find(banned.begin(), banned.end(), addr); if (it != banned.end()) banned.erase(it); }