SQCSimulator2023/SQCSim-srv/server.cpp

270 lines
7.0 KiB
C++
Raw Normal View History

2023-09-24 08:45:40 -04:00
#include "server.h"
2023-09-24 11:07:03 -04:00
Server::Server(LogDest log) {
m_log = log;
if (log == LOG_DEST::LOGFILE) {
2023-09-24 08:45:40 -04:00
m_logfile = std::ofstream("server.log", std::ofstream::out);
if (!m_logfile.is_open()) {
2023-09-24 11:07:03 -04:00
m_log = LOG_DEST::CONSOLE; // Fallback console.
2023-09-24 08:45:40 -04:00
Log("Ouverture fichier log: repli vers console.", true, false);
}
}
}
Server::~Server() {
if (m_logfile.is_open())
m_logfile.close();
if (m_sock_udp)
closesocket(m_sock_udp);
if (m_sock_tcp)
closesocket(m_sock_tcp);
#ifdef _WIN32
WSACleanup();
#endif
}
int Server::Init() {
Log("Initialisation du serveur...", false, false);
#ifdef _WIN32
if (WSAStartup(MAKEWORD(2, 2), &m_wsaData) != 0) { /* Initialisation de l'environnement reseau (Windows only) */
Log("Initialisation WinSock.", true, true);
return 1;
}
#endif
m_sock_udp = socket(AF_INET, SOCK_DGRAM, 0);
if (m_sock_udp == INVALID_SOCKET) { /* Creation du socket UDP */
Log("Creation Socket UDP.", true, true);
return 2;
}
m_sock_tcp = socket(AF_INET, SOCK_STREAM, 0);
if (m_sock_tcp == INVALID_SOCKET) { /* Creation du socket TCP */
Log("Creation Socket TCP.", true, true);
return 3;
}
/* Creation structure donnes descripteur du socket serveur */
sockaddr_in addr;
addr.sin_family = AF_INET;
addr.sin_port = htons(SRV_PORT);
addr.sin_addr.s_addr = htonl(INADDR_ANY);
if (bind(m_sock_udp, (sockaddr*)&addr, sizeof(addr)) != 0) { /* Associer le socket UDP au port */
Log("Association Socket UDP.", true, true);
return 4;
}
if (bind(m_sock_tcp, (sockaddr*)&addr, sizeof(addr)) != 0) { /* Associer le socket TCP au port */
Log("Association Socket TCP.", true, true);
return 5;
}
return 0;
}
int Server::Ready() {
2023-10-18 09:33:56 -04:00
int nbrjoueurs = 0,
nbrconn = 0;
2023-10-26 10:39:08 -04:00
char *buf = new char[BUFFER_LENGTH],
*strbuf = new char[BUFFER_LENGTH];
uint32_t buflen = BUFFER_LENGTH,
strbuflen = BUFFER_LENGTH;
bool readystart = false;
2023-10-18 09:33:56 -04:00
do {
Log("Entrez la dur<75>e de la partie: ", false, false);
std::cin >> m_game.countdown;
std::cout << std::endl;
} while (m_game.countdown < 1);
do {
Log("Entrez le seed de la partie: ", false, false);
std::cin >> m_game.seed;
std::cout << std::endl;
} while (m_game.seed < 1);
do {
Log("Entrez le nombre de joueurs: ", false, false);
std::cin >> nbrjoueurs;
std::cout << std::endl;
} while (nbrjoueurs > 0 && nbrjoueurs >= MAX_CONNECTIONS);
2023-10-18 09:33:56 -04:00
m_game.gameType = 1;
2023-09-24 11:07:03 -04:00
if (listen(m_sock_tcp, MAX_CONNECTIONS) < 0) {
Log("<EFBFBD>coute sur le port TCP.", true, true);
return 1;
}
srand(time(NULL));
2023-09-28 09:15:39 -04:00
buildIdList(ID_LIST_SIZE);
2023-10-26 10:39:08 -04:00
Log("<EFBFBD> l'<27>coute sur le port: " + std::to_string(SRV_PORT), false, false);
2023-09-24 11:07:03 -04:00
while (!readystart) {
2023-09-27 11:24:41 -04:00
sockaddr_in sockad;
2023-10-24 07:32:16 -04:00
addrlen_t addrlen = sizeof(sockad);
2023-09-27 11:24:41 -04:00
SOCKET sock = accept(m_sock_tcp, (sockaddr*)&sockad, &addrlen);
if (sock < 0)
Log("Erreur de connexion", true, false);
else if (sock > 0) {
std::string str = "Nouvelle connection provenant de: ";
str.append(inet_ntop(AF_INET, &sockad.sin_addr, strbuf, strbuflen)).append(": ").append(std::to_string(sockad.sin_port));
2023-09-27 11:24:41 -04:00
if (recv(sock, buf, buflen, 0) > 0) {
2023-10-25 12:16:14 -04:00
LoginInfo* log;
PlayerInfo play;
Packet pck = getPack(buf, buflen);
if (pck.type != PACKET_TYPE::LOGINF) {
Log("Paquet invalide.", true, false);
if (pck.type != PACKET_TYPE::ERR)
emptyPack(pck);
continue; // Passer au prochain appel si c'est pas un LoginInfo ou un LoginInfo invalide qui rentre.
}
2023-10-25 12:16:14 -04:00
log = (LoginInfo*)pck.ptr;
log->sid = getUniqueId();
log->tid = 0;
2023-09-28 09:15:39 -04:00
2023-10-25 12:16:14 -04:00
Log(str.append(" Nom: ").append(log->name), false, false);
str.clear();
2023-09-27 11:24:41 -04:00
2023-10-25 12:16:14 -04:00
Log(str.append(log->name).append(" SID: [").append(std::to_string(log->sid).append("]")), false, false);
2023-09-28 09:15:39 -04:00
2023-10-25 12:16:14 -04:00
sendPack<LoginInfo>(sock, log, &buf, &buflen);
play.id = getUniqueId();
2023-10-26 10:39:08 -04:00
strcpy(play.name, log->name);
2023-10-25 12:16:14 -04:00
play.tid = log->tid;
2023-09-28 09:15:39 -04:00
2023-10-25 12:16:14 -04:00
sendPack<GameInfo>(sock, &m_game, &buf, &buflen);
Connection* conn = new Connection(sock, sockad, *log, play);
2023-09-28 09:15:39 -04:00
2023-10-26 10:39:08 -04:00
for (auto& [key, player] : m_players) {
sendPack<PlayerInfo>(player->getSock(), &play, &buf, &buflen); // Envoyer les infos de joueur distant aux joueurs d<>j<EFBFBD> connect<63>s
2023-10-25 12:16:14 -04:00
buflen = BUFFER_LENGTH;
2023-10-26 10:39:08 -04:00
sendPack<PlayerInfo>(sock, player->getInfo(), &buf, &buflen); // et envoyer les infos des joueurs distants au nouveau joueur.
2023-10-25 12:16:14 -04:00
buflen = BUFFER_LENGTH;
}
m_players[log->sid] = conn;
2023-10-25 12:16:14 -04:00
delete log; // le pck va se supprimer tout seul, mais le pointer du log qui vient de lui, non.
if (++nbrconn >= nbrjoueurs)
readystart = true;
2023-09-27 11:24:41 -04:00
}
}
2023-09-24 11:07:03 -04:00
}
2023-10-26 10:39:08 -04:00
delete buf;
delete strbuf;
2023-09-24 08:45:40 -04:00
return 0;
}
void Server::Run() {
2023-10-23 16:21:15 -04:00
char* buf = new char[BUFFER_LENGTH];
2023-10-23 16:54:09 -04:00
uint32_t buflen = BUFFER_LENGTH;
netprot::Input in;
2023-10-23 17:49:40 -04:00
sockaddr_in sockad;
2023-10-26 10:39:08 -04:00
addrlen_t socklen = sizeof(sockad);
2023-10-23 17:49:40 -04:00
2023-10-25 12:16:14 -04:00
Log("Debut de la partie...", false, false);
2023-10-23 16:21:15 -04:00
2023-10-26 10:39:08 -04:00
for (auto& [key, conn]: m_players) // G<>rer le point de spawn des joueurs.
conn->player = new Player(Vector3f(64., 128., 64.));
2023-10-25 12:16:14 -04:00
// TODO: Faire un premier sync pour que les joueurs partent <20> la bonne place.
2023-10-23 16:21:15 -04:00
while (true) {
2023-10-23 17:49:40 -04:00
if (recvfrom(m_sock_udp, buf, BUFFER_LENGTH, 0, (sockaddr*)&sockad, &socklen) > 0) {
2023-10-23 16:54:09 -04:00
Deserialize(&in, buf, buflen);
std::cout << "Id: " << in.sid << "\r\n"
<< "Direction: { " << in.direction.x << ", " << in.direction.y << ", " << in.direction.z << " }" << "\r\n";
2023-10-23 16:21:15 -04:00
}
2023-10-23 16:54:09 -04:00
std::cout << "!" << std::endl;
2023-10-23 16:21:15 -04:00
}
2023-09-24 08:45:40 -04:00
}
inline std::string Server::LogTimestamp() {
2023-09-24 08:45:40 -04:00
time_t rawtime;
struct tm timeinfo;
char buffer[80];
time(&rawtime);
#ifdef _WIN32
localtime_s(&timeinfo, &rawtime);
#else
localtime_r(&rawtime, &timeinfo);
#endif
strftime(buffer, sizeof(buffer), "%d-%m-%Y %H:%M:%S", &timeinfo);
std::string str(buffer);
return "[" + str + "] ";
}
void Server::Log(std::string str, bool is_error = false, bool is_fatal = false) {
switch (m_log) {
2023-10-26 10:39:08 -04:00
using enum LOG_DEST; // C++20!
case LOGFILE:
m_logfile << LogTimestamp() << (is_fatal ? "FATAL " : "") << (is_error ? "ERROR " : "") << str << std::endl;
2023-09-24 08:45:40 -04:00
break;
2023-10-26 10:39:08 -04:00
case CONSOLE:
2023-09-24 08:45:40 -04:00
default:
std::cout << LogTimestamp() << (is_fatal ? "FATAL " : "") << (is_error ? "ERROR " : "") << str << std::endl;
2023-09-24 08:45:40 -04:00
break;
}
if (is_fatal) {
if (m_sock_udp)
closesocket(m_sock_udp);
if (m_sock_tcp)
closesocket(m_sock_tcp);
2023-10-26 10:39:08 -04:00
for (auto& [key, player] : m_players)
closesocket(player->getSock());
m_players.clear();
2023-09-24 08:45:40 -04:00
#ifdef _WIN32
WSACleanup();
#endif
}
}
2023-09-28 09:15:39 -04:00
void Server::buildIdList(size_t size) {
std::set<uint64_t> lst;
2023-10-24 07:32:16 -04:00
do lst.insert(((uint64_t)rand() << 32 | rand())); // EIGHT SIX SEVENFIVE THREE AUGHT NIIIIIIiIIiiIiINE!
2023-09-28 09:15:39 -04:00
while (lst.size() < size);
m_ids = std::vector<uint64_t>(lst.begin(), lst.end());
}
uint64_t Server::getUniqueId() {
uint64_t id = m_ids.back();
m_ids.pop_back();
return id;
}
2023-09-25 16:30:03 -04:00
// Test serialize/deserialize:
/*
2023-09-27 11:24:41 -04:00
netprot::LoginInfo log, log2;
std::cout << "Nom? ";
std::cin.getline(log.name, 32); // NO! STD::CIN >> VARIABLE;! EVEEEEEERRRR!!!
log.sid = 12345;
2023-09-25 16:30:03 -04:00
char* buf = new char[150];
uint32_t buflen = 150;
2023-09-27 11:24:41 -04:00
netprot::Serialize(&log, &buf, &buflen);
2023-09-25 16:30:03 -04:00
2023-09-27 11:24:41 -04:00
bool is_work = netprot::Deserialize(&log2, buf, buflen);
2023-09-25 16:30:03 -04:00
std::string str;
2023-09-27 11:24:41 -04:00
str.append(is_work ? "Y " : "N ").append(log2.name).append(": ").append(std::to_string(log2.sid));
2023-09-25 16:30:03 -04:00
Log(str, false, false);
*/