OpenFusion/src/CNShardServer.cpp

118 lines
3.8 KiB
C++
Raw Normal View History

2020-08-18 20:42:30 +00:00
#include "CNProtocol.hpp"
#include "CNStructs.hpp"
#include "CNShardServer.hpp"
#include "PlayerManager.hpp"
#include "MobManager.hpp"
2020-08-18 20:42:30 +00:00
#include "CNShared.hpp"
2020-08-22 17:39:13 +00:00
#include "settings.hpp"
#include "Database.hpp"
#include "TableData.hpp" // for flush()
2020-08-18 20:42:30 +00:00
#include <iostream>
#include <sstream>
#include <cstdlib>
std::map<uint32_t, PacketHandler> CNShardServer::ShardPackets;
std::list<TimerEvent> CNShardServer::Timers;
2020-08-18 20:42:30 +00:00
CNShardServer::CNShardServer(uint16_t p) {
port = p;
pHandler = &CNShardServer::handlePacket;
REGISTER_SHARD_TIMER(keepAliveTimer, 4000);
REGISTER_SHARD_TIMER(periodicSaveTimer, settings::DBSAVEINTERVAL*1000);
2020-08-18 20:42:30 +00:00
init();
}
void CNShardServer::handlePacket(CNSocket* sock, CNPacketData* data) {
2020-08-23 21:09:31 +00:00
printPacket(data, CL2FE);
2020-08-22 17:39:13 +00:00
2020-08-18 20:42:30 +00:00
if (ShardPackets.find(data->type) != ShardPackets.end())
ShardPackets[data->type](sock, data);
2020-08-31 22:54:49 +00:00
else if (settings::VERBOSITY > 0)
std::cerr << "OpenFusion: SHARD UNIMPLM ERR. PacketType: " << Defines::p2str(CL2FE, data->type) << " (" << data->type << ")" << std::endl;
PlayerManager::players[sock].lastHeartbeat = getTime();
2020-08-18 20:42:30 +00:00
}
void CNShardServer::keepAliveTimer(CNServer* serv, time_t currTime) {
2020-09-18 02:41:09 +00:00
for (auto& pair : PlayerManager::players) {
if (pair.second.lastHeartbeat != 0 && currTime - pair.second.lastHeartbeat > settings::TIMEOUT) {
// if the client hasn't responded in 60 seconds, its a dead connection so throw it out
pair.first->kill();
} else if (pair.second.lastHeartbeat != 0 && currTime - pair.second.lastHeartbeat > settings::TIMEOUT/2) {
// if the player hasn't responded in 30 seconds, send a live check
INITSTRUCT(sP_FE2CL_REQ_LIVE_CHECK, data);
pair.first->sendPacket((void*)&data, P_FE2CL_REQ_LIVE_CHECK, sizeof(sP_FE2CL_REQ_LIVE_CHECK));
}
}
}
void CNShardServer::periodicSaveTimer(CNServer* serv, time_t currTime) {
if (PlayerManager::players.empty())
return;
std::cout << "[INFO] Saving " << PlayerManager::players.size() << " players to DB..." << std::endl;
2020-09-18 02:41:09 +00:00
for (auto& pair : PlayerManager::players) {
Database::updatePlayer(pair.second.plr);
}
TableData::flush();
std::cout << "[INFO] Done." << std::endl;
}
2020-08-22 23:31:09 +00:00
void CNShardServer::newConnection(CNSocket* cns) {
cns->setActiveKey(SOCKETKEY_E); // by default they accept keys encrypted with the default key
}
// must be static to be called from PlayerManager::exitDuplicate()
void CNShardServer::_killConnection(CNSocket* cns) {
// check if the player ever sent a REQ_PC_ENTER
if (PlayerManager::players.find(cns) == PlayerManager::players.end())
return;
2020-09-28 18:11:13 +00:00
Player* plr = PlayerManager::getPlayer(cns);
if (plr == nullptr) { // this shouldn't happen if everything works correctly...
PlayerManager::removePlayer(cns);
// also, hopefully the player's progress was already saved since the last db save interval, but rip those 2 mins of progress lol
2020-09-28 18:11:13 +00:00
return;
}
2020-09-28 18:11:13 +00:00
int64_t key = plr->SerialKey;
PlayerManager::removePlayer(cns); // removes the player from the list and saves it to DB
2020-08-18 20:42:30 +00:00
2020-09-18 02:41:09 +00:00
// remove from CNSharedData
CNSharedData::erasePlayer(key);
2020-08-19 01:34:39 +00:00
}
void CNShardServer::killConnection(CNSocket *cns) {
_killConnection(cns);
}
// flush the DB when terminating the server
void CNShardServer::kill() {
periodicSaveTimer(nullptr, 0);
CNServer::kill();
}
void CNShardServer::onStep() {
time_t currTime = getTime();
2020-08-19 01:34:39 +00:00
for (TimerEvent& event : Timers) {
if (event.scheduledEvent == 0) {
// event hasn't been queued yet, go ahead and do that
event.scheduledEvent = currTime + event.delta;
2020-08-19 01:34:39 +00:00
continue;
}
if (event.scheduledEvent < currTime) {
// timer needs to be called
event.handlr(this, currTime);
event.scheduledEvent = currTime + event.delta;
}
2020-08-19 01:34:39 +00:00
}
}