forked from mirror/openmw-tes3mp
333 lines
12 KiB
C++
333 lines
12 KiB
C++
//
|
|
// Created by koncord on 21.04.17.
|
|
//
|
|
|
|
#include <RakPeerInterface.h>
|
|
#include <RakSleep.h>
|
|
#include <BitStream.h>
|
|
#include <iostream>
|
|
#include "MasterServer.hpp"
|
|
|
|
#include <components/openmw-mp/Master/PacketMasterQuery.hpp>
|
|
#include <components/openmw-mp/Master/PacketMasterUpdate.hpp>
|
|
#include <components/openmw-mp/Master/PacketMasterAnnounce.hpp>
|
|
#include <components/openmw-mp/Version.hpp>
|
|
#include <components/openmw-mp/Utils.hpp>
|
|
#include <boost/filesystem.hpp>
|
|
|
|
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<MasterServer::SServer>("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<unsigned short>(), nullptr);
|
|
peer->Startup(maxConnections.as<unsigned short>(), &sockdescr, 1, 1000);
|
|
peer->SetLimitIPConnectionFrequency(true);
|
|
|
|
peer->SetMaximumIncomingConnections(maxConnections.as<unsigned short>());
|
|
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);
|
|
|
|
luaStuff([](sol::state &state) {
|
|
sol::protected_function func = state["OnInit"];
|
|
sol::protected_function_result result = func.call();
|
|
if (!result.valid())
|
|
cerr << "Error: " << result.get<string>() << endl;
|
|
});
|
|
|
|
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<map<SystemAddress, QueryData> *>(&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
|
|
|
|
auto it = servers.find(addr);
|
|
if (it != servers.end())
|
|
{
|
|
pair<SystemAddress, QueryData> pairPtr(it->first, static_cast<QueryData>(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:
|
|
{
|
|
auto 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);
|
|
|
|
lock_guard<mutex> lock(banMutex);
|
|
|
|
if (peer->IsBanned(addr)) // 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<bool>();
|
|
else
|
|
cerr << "Error: " << result.get<string>() << 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)
|
|
{
|
|
|
|
if (isServerValid(server))
|
|
{
|
|
cout << "Updated";
|
|
iter->second = server;
|
|
keepAliveFunc();
|
|
}
|
|
else
|
|
{
|
|
cout << "Update rejected";
|
|
servers.erase(iter);
|
|
pendingACKs.erase(packet->guid);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
cout << "Keeping alive";
|
|
keepAliveFunc();
|
|
}
|
|
}
|
|
else if (pma.GetFunc() == PacketMasterAnnounce::FUNCTION_ANNOUNCE)
|
|
{
|
|
if (isServerValid(server))
|
|
{
|
|
cout << "Added";
|
|
iter = servers.insert({packet->systemAddress, server}).first;
|
|
keepAliveFunc();
|
|
}
|
|
else
|
|
cout << "Adding rejected";
|
|
}
|
|
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<void(sol::state &)> f)
|
|
{
|
|
lock_guard<mutex> lock(luaMutex);
|
|
f(state);
|
|
}
|
|
|
|
void MasterServer::ban(const std::string &addr)
|
|
{
|
|
lock_guard<mutex> lock(banMutex);
|
|
peer->AddToBanList(addr.c_str());
|
|
}
|
|
|
|
void MasterServer::unban(const std::string &addr)
|
|
{
|
|
lock_guard<mutex> lock(banMutex);
|
|
peer->RemoveFromBanList(addr.c_str());
|
|
}
|