mirror of
https://git.do.srb2.org/STJr/SRB2.git
synced 2024-12-22 18:51:33 +00:00
1041 lines
31 KiB
C++
1041 lines
31 KiB
C++
// Emacs style mode select -*- C++ -*-
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
// Copyright (C) 2000 by DooM Legacy Team.
|
|
//
|
|
// This program is free software; you can redistribute it and/or
|
|
// modify it under the terms of the GNU General Public License
|
|
// as published by the Free Software Foundation; either version 2
|
|
// of the License, or (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
#ifdef __GNUC__
|
|
#include <unistd.h>
|
|
#include <sys/types.h>
|
|
#endif
|
|
#include <typeinfo>
|
|
#include <string.h>
|
|
#include <stdarg.h>
|
|
#include "ipcs.h"
|
|
#include "common.h"
|
|
#include "mysql.h"
|
|
#include "md5.h"
|
|
#include <time.h>
|
|
//#include <netdb.h>
|
|
|
|
//=============================================================================
|
|
|
|
#ifdef __GNUC__
|
|
#define ATTRPACK __attribute__ ((packed))
|
|
#else
|
|
#define ATTRPACK
|
|
#endif
|
|
|
|
#define PT_ASKINFOVIAMS 15
|
|
|
|
static CServerSocket server_socket;
|
|
|
|
FILE *logfile;
|
|
FILE *errorfile;
|
|
FILE *mysqlfile;
|
|
FILE *sockfile;
|
|
FILE *pidfile;
|
|
|
|
#if defined(_MSC_VER)
|
|
#pragma pack(1)
|
|
#endif
|
|
|
|
typedef struct
|
|
{
|
|
char ip[16]; // Big enough to hold a full address.
|
|
UINT16 port;
|
|
UINT8 padding1[2];
|
|
UINT32 time;
|
|
} ATTRPACK ms_holepunch_packet_t;
|
|
|
|
typedef struct
|
|
{
|
|
char clientaddr[22];
|
|
UINT8 padding1[2];
|
|
UINT32 time;
|
|
} ATTRPACK msaskinfo_pak;
|
|
|
|
//
|
|
// SRB2 network packet data.
|
|
//
|
|
typedef struct
|
|
{
|
|
UINT32 checksum;
|
|
UINT8 ack; // if not null the node asks for acknowledgement, the receiver must resend the ack
|
|
UINT8 ackreturn; // the return of the ack number
|
|
|
|
UINT8 packettype;
|
|
UINT8 reserved; // padding
|
|
|
|
msaskinfo_pak msaskinfo;
|
|
} ATTRPACK doomdata_t;
|
|
|
|
#if defined(_MSC_VER)
|
|
#pragma pack()
|
|
#endif
|
|
|
|
//=============================================================================
|
|
|
|
#define HOSTNAME "localhost"
|
|
#define USER "srb2_ms"
|
|
#define PASSWORD "gLRDRb7WgLRDRb7W"
|
|
#define DATABASE "srb2_ms"
|
|
|
|
// MySQL Stuff :D
|
|
|
|
const char *server = HOSTNAME;
|
|
const char *user = USER;
|
|
const char *password = PASSWORD;
|
|
const char *database = DATABASE;
|
|
time_t lastupdate;
|
|
|
|
MYSQL *conn;
|
|
MYSQL_RES *res;
|
|
MYSQL_ROW row;
|
|
int mysqlconnected = 0;
|
|
/*
|
|
char *str_replace(char * t1, char * t2, char * t6)
|
|
{
|
|
char*t4;
|
|
char*t5=new(0);
|
|
|
|
while(strstr(t6,t1)){
|
|
t4=strstr(t6,t1);
|
|
strncpy(t5+strlen(t5),t6,t4-t6);
|
|
strcat(t5,t2);
|
|
t4+=strlen(t1);
|
|
t6=t4;
|
|
}
|
|
return strcat(t5,t4);
|
|
}
|
|
*/
|
|
/*
|
|
// Cue was here
|
|
char *str_quakeformat(char *msg)
|
|
{
|
|
char *quakemsg , *quakemsg1, *quakemsg2;
|
|
char *quakemsg3, *quakemsg4, *quakemsg5;
|
|
char *quakemsg6, *quakemsg7, *quakemsg8;
|
|
|
|
quakemsg1 = new(sizeof(char) * (strlen(msg) + 1));
|
|
strcpy(quakemsg1, msg);
|
|
quakemsg2 = str_replace("^1", "\x81",quakemsg1);
|
|
quakemsg3 = str_replace("^2", "\x82",quakemsg2);
|
|
quakemsg4 = str_replace("^3", "\x83",quakemsg3);
|
|
quakemsg5 = str_replace("^4", "\x84",quakemsg4);
|
|
quakemsg6 = str_replace("^5", "\x85",quakemsg5);
|
|
quakemsg7 = str_replace("^6", "\x86",quakemsg6);
|
|
quakemsg8 = str_replace("^7", "\x87",quakemsg7);
|
|
quakemsg = str_replace("^0", "\x80",quakemsg8);
|
|
delete(quakemsg1);
|
|
delete(quakemsg2);
|
|
delete(quakemsg3);
|
|
delete(quakemsg4);
|
|
delete(quakemsg5);
|
|
delete(quakemsg6);
|
|
delete(quakemsg7);
|
|
delete(quakemsg8);
|
|
|
|
return quakemsg;
|
|
}
|
|
*/
|
|
void MySQL_Conn(bool force) {
|
|
//if(mysql_ping(conn))
|
|
//mysqlconnected = 0;
|
|
logPrintf(mysqlfile, "Trying to connect to MySQL...\n");
|
|
if(mysqlconnected != 1 || force) {
|
|
if(force)
|
|
mysql_close(conn);
|
|
const char *server = HOSTNAME;
|
|
const char *user = USER;
|
|
const char *password = PASSWORD;
|
|
const char *database = DATABASE;
|
|
|
|
conn = mysql_init(NULL);
|
|
|
|
/* Connect to database */
|
|
if (!mysql_real_connect(conn, server,
|
|
user, password, database, 8219, NULL, 0)) {
|
|
logPrintf(errorfile, "%s\n", mysql_error(conn));
|
|
}
|
|
conn = mysql_init(NULL);
|
|
|
|
/* Connect to database */
|
|
if (!mysql_real_connect(conn, server, user, password, database, 0, NULL, 0)) {
|
|
logPrintf(errorfile, "%s\n", mysql_error(conn));
|
|
mysqlconnected = 0;
|
|
} else {
|
|
logPrintf(mysqlfile, "Connection to MySQL Database successful!\n");
|
|
mysqlconnected = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
static int MySQL_CheckBan(const char *ip, UINT32 id, bool sendinfo, bool type) {
|
|
msg_t msg;
|
|
int writecode;
|
|
MySQL_Conn(false);
|
|
msg.id = id;
|
|
char banqueryp1[500] = "SELECT bid,INET_NTOA(ipstart),INET_NTOA(ipend),DATE_FORMAT(FROM_UNIXTIME(full_endtime),'%%a %%b %%e %%Y at %%k:%%i (CDT)'),reason,hostonly,permanent FROM ms_bans WHERE INET_ATON('%s') BETWEEN `ipstart` AND `ipend` AND (`full_endtime` > %ld OR `permanent` = '1') LIMIT 1";
|
|
// char exqueryp1[500] = "SELECT * FROM ms_exceptions WHERE `ip` = '%s' AND `bid` = '%s' LIMIT 1";
|
|
// char exquery[500];
|
|
char banquery[500];
|
|
// char bid[5];
|
|
|
|
time_t current_time = time (NULL);
|
|
sprintf(banquery, banqueryp1, ip, current_time);
|
|
logPrintf(mysqlfile, "Executing MySQL Query: %s\n", banquery);
|
|
if(mysql_query(conn, banquery)) {
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
return false;
|
|
} else {
|
|
res = mysql_store_result(conn);
|
|
if(mysql_num_rows(res) >= 1) {
|
|
logPrintf(logfile, "We got a ban coming on!\n");
|
|
row = mysql_fetch_row(res);
|
|
if(!type && strcmp(row[5],"1") == 0)
|
|
return false;
|
|
if(sendinfo)
|
|
{
|
|
logPrintf(logfile, "And we're meant to tell them some more!\n");
|
|
msg_ban_t *info = (msg_ban_t *) msg.buffer;
|
|
info->header[0] = '\0';
|
|
|
|
logPrintf(logfile, "Got Ban: %s - %s, ends %s, for %s\n", row[1], row[2], row[3], row[4]);
|
|
logPrintf(logfile, "COPY: ipstart\n");
|
|
strcpy(info->ipstart, row[1]);
|
|
logPrintf(logfile, "COPY: ipend\n");
|
|
strcpy(info->ipend, row[2]);
|
|
logPrintf(logfile, "COPY: endtime\n");
|
|
if(strcmp(row[6], "0") == 0)
|
|
{
|
|
logPrintf(logfile, "COPY: It's not permanent.\n");
|
|
strcpy(info->endstamp, row[3]);
|
|
}
|
|
else
|
|
{
|
|
logPrintf(logfile, "COPY: It's permanent.\n");
|
|
strcpy(info->endstamp, "Never");
|
|
}
|
|
logPrintf(logfile, "COPY: reason\n");
|
|
strcpy(info->reason, row[4]);
|
|
logPrintf(logfile, "COPY: hostonly\n");
|
|
if(strcmp(row[5],"0") == 0)
|
|
info->hostonly = false;
|
|
else
|
|
info->hostonly = true;
|
|
|
|
if(info->hostonly)
|
|
logPrintf(logfile, "BAN HAMMER: %s - %s || Reason: %s || Endstamp: %s || HOSTONLY\n", info->ipstart, info->ipend, info->reason, info->endstamp);
|
|
else
|
|
logPrintf(logfile, "BAN HAMMER: %s - %s || Reason: %s || Endstamp: %s || NOTHOSTONLY\n", info->ipstart, info->ipend, info->reason, info->endstamp);
|
|
mysql_free_result(res);
|
|
msg.type = GET_BANNED_MSG;
|
|
msg.length = sizeof (msg_ban_t);
|
|
msg.room = 0;
|
|
logPrintf(logfile, "Sending message?\n");
|
|
writecode = server_socket.write(&msg);
|
|
logPrintf(logfile, "Message sent! :D\n");
|
|
if (writecode < 0)
|
|
{
|
|
dbgPrintf(LIGHTRED, "Write error... %d client %d "
|
|
"deleted\n", writecode, id);
|
|
return true;
|
|
}
|
|
}
|
|
return true;
|
|
} else {
|
|
mysql_free_result(res);
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
int MySQL_CheckRoom(UINT32 room)
|
|
{
|
|
MySQL_Conn(false);
|
|
char checkqueryp1[500] = "SELECT private FROM `ms_rooms` WHERE `room_id` = '%d' AND `private` = '1' LIMIT 1";
|
|
char checkquery[500];
|
|
sprintf(checkquery, checkqueryp1, room);
|
|
logPrintf(mysqlfile, "Executing MySQL Query: %s\n", checkquery);
|
|
if(mysql_query(conn, checkquery)) {
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
return false;
|
|
} else {
|
|
res = mysql_store_result(conn);
|
|
if(mysql_num_rows(res) < 1)
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void MySQL_AddServer(const char *ip, const char *port, const char *name, const char *version, UINT32 room, bool firstadd, const char *key) {
|
|
char escapedName[255];
|
|
char escapedPort[10];
|
|
char escapedVersion[10];
|
|
char escapedKey[32];
|
|
char insertquery[5000];
|
|
char checkquery[500];
|
|
char updatequery[5000];
|
|
char queryp1[5000] = "INSERT INTO `ms_servers` (`name`,`ip`,`port`,`version`,`timestamp`,`room`,`key`) VALUES ('%s','%s','%s','%s','%ld','%d','%s')";
|
|
char checkqueryp1[500] = "SELECT room_override FROM `ms_servers` WHERE `ip` = '%s' AND `port` = '%s'";
|
|
char updatequeryp1[5000];
|
|
if(firstadd)
|
|
{
|
|
logPrintf(logfile, "First add.\n");
|
|
strcpy(updatequeryp1, "UPDATE `ms_servers` SET `name` = '%s', `port` = '%s', `version` = '%s', timestamp = '%ld', upnow = '1', `room` = '%d', `delisted` = '0', `key` = '%s' WHERE `ip` = '%s' AND `port` = '%s'");
|
|
}
|
|
else
|
|
{
|
|
logPrintf(logfile, "Update ping.\n");
|
|
strcpy(updatequeryp1, "UPDATE `ms_servers` SET `name` = '%s', `port` = '%s', `version` = '%s', timestamp = '%ld', upnow = '1', `room` = '%d', `key` = '%s' WHERE `ip` = '%s' AND `port` = '%s' AND `delisted` = '0'");
|
|
}
|
|
MySQL_Conn(false);
|
|
mysql_real_escape_string(conn, escapedName, name, (unsigned long)strlen(name));
|
|
mysql_real_escape_string(conn, escapedPort, port, (unsigned long)strlen(port));
|
|
mysql_real_escape_string(conn, escapedVersion, version, (unsigned long)strlen(version));
|
|
mysql_real_escape_string(conn, escapedKey, key, (unsigned long)strlen(key));
|
|
if(!MySQL_CheckBan(ip,0,false,1)) {
|
|
if(!MySQL_CheckRoom(room))
|
|
{
|
|
logPrintf(errorfile, "IP %s tried to use the private room %d! THIS SHOULD NOT HAPPEN\n", ip, room);
|
|
return;
|
|
}
|
|
sprintf(checkquery, checkqueryp1, ip, port);
|
|
time_t timestamp;
|
|
timestamp = time (NULL);
|
|
logPrintf(logfile, "Checking for existing servers in table with the same IP and port...\n");
|
|
logPrintf(mysqlfile, "Executing MySQL Query: %s\n", checkquery);
|
|
if(mysql_query(conn, checkquery)) {
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
} else {
|
|
res = mysql_store_result(conn);
|
|
logPrintf(logfile, "Found %d rows...\n", mysql_num_rows(res));
|
|
if(mysql_num_rows(res) < 1) {
|
|
mysql_free_result(res);
|
|
logPrintf(logfile, "Adding the temporary server: %s:%s || Name: %s || Version: %s || Time: %ld || Room: %d || Key: %s\n", ip, port, name, version, timestamp, room, key);
|
|
sprintf(insertquery, queryp1, escapedName, ip, escapedPort, escapedVersion, timestamp, room, escapedKey);
|
|
logPrintf(mysqlfile, "Executing MySQL Query: %s\n", insertquery);
|
|
if(mysql_query(conn, insertquery)) {
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
} else {
|
|
logPrintf(logfile, "Server added successfully!\n");
|
|
}
|
|
} else {
|
|
row = mysql_fetch_row(res);
|
|
if(atoi(row[0]) != 0)
|
|
room = atoi(row[0]);
|
|
mysql_free_result(res);
|
|
logPrintf(logfile, "Server's IP and port already exists, so let's just update it instead...\n");
|
|
logPrintf(logfile, "Updating Server Data for %s\n", ip);
|
|
sprintf(updatequery, updatequeryp1, escapedName, escapedPort, escapedVersion, timestamp, room, escapedKey, ip, port);
|
|
logPrintf(mysqlfile, "Executing MySQL Query: %s\n", updatequery);
|
|
if(mysql_query(conn, updatequery)) {
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
} else {
|
|
logPrintf(logfile, "Server status for Server: %s:%s || Name: %s || Version: %s || Time: %ld || Room: %d || Key: %s set successfully.\n", ip, port, name, version, timestamp, room, key);
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
logPrintf(logfile, "IP %s is banned so do nothing.\n", ip);
|
|
}
|
|
}
|
|
|
|
void MySQL_ListServers(UINT32 id, UINT32 type, const char *ip, UINT32 room) {
|
|
msg_t msg;
|
|
int writecode;
|
|
MySQL_Conn(false);
|
|
msg.id = id;
|
|
msg.type = type;
|
|
char servquery[1000];
|
|
if(room == 0)
|
|
sprintf(servquery, "SELECT ip, port, name, version FROM ( SELECT * FROM `ms_servers` WHERE `upnow` = '1' ORDER BY `sid` ASC) as t2 ORDER BY `sticky` DESC");
|
|
else
|
|
{
|
|
char servqueryp1[1000] = "SELECT ip, port, name, version FROM ( SELECT * FROM `ms_servers` WHERE `upnow` = '1' AND `room` = '%d' ORDER BY `sid` ASC) as t2 ORDER BY `sticky` DESC";
|
|
sprintf(servquery, servqueryp1, room);
|
|
}
|
|
if(!MySQL_CheckBan(ip,0,false,0)) {
|
|
logPrintf(mysqlfile, "Executing MySQL Query: %s\n", servquery);
|
|
if(mysql_query(conn, servquery)) {
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
} else {
|
|
res = mysql_store_result(conn);
|
|
logPrintf(logfile, "Found %d servers...\n", mysql_num_rows(res));
|
|
while ((row = mysql_fetch_row(res)) != NULL)
|
|
{
|
|
msg_server_t *info = (msg_server_t *) msg.buffer;
|
|
|
|
info->header[0] = '\0'; // nothing interresting in it (for now)
|
|
strcpy(info->ip, row[0]);
|
|
strcpy(info->port, row[1]);
|
|
strcpy(info->name, row[2]);
|
|
strcpy(info->version, row[3]);
|
|
|
|
msg.length = sizeof (msg_server_t);
|
|
msg.room = 0;
|
|
logPrintf(logfile, "Sending message?\n");
|
|
writecode = server_socket.write(&msg);
|
|
logPrintf(logfile, "Message sent! :D\n");
|
|
if (writecode < 0)
|
|
{
|
|
dbgPrintf(LIGHTRED, "Write error... %d client %d "
|
|
"deleted\n", writecode, id);
|
|
return;
|
|
}
|
|
}
|
|
mysql_free_result(res);
|
|
}
|
|
} else {
|
|
logPrintf(logfile, "IP %s is banned so do nothing.\n", ip);
|
|
}
|
|
}
|
|
|
|
void MySQL_ListRooms(UINT32 id, UINT32 type, const char *ip, bool sendtype) {
|
|
msg_t msg;
|
|
int writecode;
|
|
//(void)ip;
|
|
MySQL_Conn(false);
|
|
msg.id = id;
|
|
msg.type = type;
|
|
char roomquery[1000];
|
|
logPrintf(logfile, "Check their ban status first... (ID %d)\n",id);
|
|
if(MySQL_CheckBan(ip,id,true,sendtype))
|
|
return;
|
|
|
|
//Hmph! I suppose it'll be up to me to handle this. ~Inuyasha
|
|
if (sendtype)
|
|
{
|
|
char overridequery[500];
|
|
|
|
sprintf(overridequery, "SELECT room_override FROM `ms_servers` WHERE `ip` = '%s' AND `room_override` > 0", ip);
|
|
logPrintf(mysqlfile, "Checking for room overrides! Executing MySQL Query: %s\n", overridequery);
|
|
if (mysql_query(conn, overridequery))
|
|
{
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
return;
|
|
}
|
|
|
|
res = mysql_store_result(conn);
|
|
if (mysql_num_rows(res) >= 1)
|
|
{
|
|
int overriddenRoom;
|
|
|
|
row = mysql_fetch_row(res);
|
|
overriddenRoom = atoi(row[0]);
|
|
mysql_free_result(res);
|
|
|
|
sprintf(roomquery, "SELECT room_id, title, motd FROM `ms_rooms` WHERE `room_id` = '%d' LIMIT 1", overriddenRoom);
|
|
if (mysql_query(conn, roomquery))
|
|
{
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
return;
|
|
}
|
|
res = mysql_store_result(conn);
|
|
if (mysql_num_rows(res) >= 1)
|
|
{
|
|
msg_rooms_t *info = (msg_rooms_t *) msg.buffer;
|
|
char roommotd[300];
|
|
|
|
row = mysql_fetch_row(res);
|
|
|
|
info->header[0] = '\0'; // nothing interesting in it (for now)
|
|
strcpy(info->name, row[1]);
|
|
|
|
// Just in case, let's make sure we don't overflow.
|
|
sprintf(roommotd, "\x85Your server has had a room override applied to it, so this is the only room you may host in.\x80\n\n%s", row[2]);
|
|
strncpy(info->motd, roommotd, 255);
|
|
info->motd[255] = '\0';
|
|
|
|
info->id = atoi(row[0]);
|
|
|
|
logPrintf(logfile, "Sending Room %s with ID %d and MOTD %s\n", info->name, info->id, info->motd);
|
|
|
|
msg.length = sizeof (msg_rooms_t);
|
|
msg.room = 0;
|
|
logPrintf(logfile, "Sending message?\n");
|
|
writecode = server_socket.write(&msg);
|
|
logPrintf(logfile, "Message sent! :D\n");
|
|
if (writecode < 0)
|
|
{
|
|
dbgPrintf(LIGHTRED, "Write error... %d client %d "
|
|
"deleted\n", writecode, id);
|
|
}
|
|
mysql_free_result(res);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
logPrintf(errorfile, "Someone's room override isn't set correctly! Room %d doesn't exist!", overriddenRoom);
|
|
mysql_free_result(res);
|
|
}
|
|
}
|
|
}
|
|
|
|
if(sendtype)
|
|
sprintf(roomquery, "SELECT room_id, title, motd FROM `ms_rooms` WHERE `private` = '0' AND `visible` = '1' ORDER BY `order` ASC");
|
|
else
|
|
sprintf(roomquery, "SELECT room_id, title, motd FROM `ms_rooms` WHERE `visible` = '1' ORDER BY `order` ASC");
|
|
if(mysql_query(conn, roomquery)) {
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
} else
|
|
{
|
|
res = mysql_store_result(conn);
|
|
logPrintf(logfile, "Found %d rooms...\n", mysql_num_rows(res));
|
|
while ((row = mysql_fetch_row(res)) != NULL)
|
|
{
|
|
msg_rooms_t *info = (msg_rooms_t *) msg.buffer;
|
|
|
|
info->header[0] = '\0'; // nothing interresting in it (for now)
|
|
strcpy(info->name, row[1]);
|
|
strcpy(info->motd, row[2]);
|
|
info->id = atoi(row[0]);
|
|
|
|
logPrintf(logfile, "Sending Room %s with ID %d and MOTD %s\n", info->name, info->id, info->motd);
|
|
|
|
msg.length = sizeof (msg_rooms_t);
|
|
msg.room = 0;
|
|
logPrintf(logfile, "Sending message?\n");
|
|
writecode = server_socket.write(&msg);
|
|
logPrintf(logfile, "Message sent! :D\n");
|
|
if (writecode < 0)
|
|
{
|
|
dbgPrintf(LIGHTRED, "Write error... %d client %d "
|
|
"deleted\n", writecode, id);
|
|
return;
|
|
}
|
|
}
|
|
mysql_free_result(res);
|
|
}
|
|
}
|
|
|
|
void MySQL_CheckVersion(UINT32 id, UINT32 type, const char *ip, UINT32 modid, const char *modversion) {
|
|
msg_t msg;
|
|
int writecode;
|
|
(void)ip;
|
|
MySQL_Conn(false);
|
|
msg.id = id;
|
|
msg.type = type;
|
|
char versionquery[1000];
|
|
sprintf(versionquery, "SELECT * FROM `ms_versions` WHERE `mod_id` = '%d' AND `mod_version` > %s LIMIT 1",modid, modversion);
|
|
logPrintf(mysqlfile, "Executing MySQL Query: %s\n", versionquery);
|
|
if(mysql_query(conn, versionquery)) {
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
} else
|
|
{
|
|
res = mysql_store_result(conn);
|
|
if(mysql_num_rows(res) < 1)
|
|
{
|
|
logPrintf(logfile, "Buffer to NULL\n");
|
|
strcpy(msg.buffer,"NULL");
|
|
}
|
|
else
|
|
{
|
|
row = mysql_fetch_row(res);
|
|
logPrintf(logfile, "Version found: %s || Codebase: %s || Version: %s || Version String: %s || Mod ID: %s\n", row[4], row[3], row[1], row[2], row[0]);
|
|
strcpy(msg.buffer,row[2]);
|
|
}
|
|
msg.length = sizeof msg.buffer;
|
|
msg.room = 0;
|
|
logPrintf(logfile, "Sending message?\n");
|
|
writecode = server_socket.write(&msg);
|
|
logPrintf(logfile, "Message sent! :D\n");
|
|
if (writecode < 0)
|
|
{
|
|
dbgPrintf(LIGHTRED, "Write error... %d client %d "
|
|
"deleted\n", writecode, id);
|
|
return;
|
|
}
|
|
mysql_free_result(res);
|
|
}
|
|
}
|
|
|
|
void MySQL_ListServServers(UINT32 id, UINT32 type, const char *ip) {
|
|
msg_t msg;
|
|
int writecode;
|
|
static char str[1024];
|
|
//char banqueryp1[500] = "SELECT reason,name,FROM_UNIXTIME(endtime) FROM ms_bans WHERE INET_ATON('%s') BETWEEN `ipstart` AND `ipend` AND `endtime` < %ld LIMIT 1";
|
|
//char banquery[500];
|
|
//time_t current_time = time (NULL);
|
|
char servquery[1000] = "SELECT ip, port, name, version, permanent FROM ( SELECT * FROM `ms_servers` WHERE `upnow` = '1' OR `permanent` = '1' ORDER BY `sid` ASC) as ms_servers ORDER BY `sticky` DESC";
|
|
MySQL_Conn(false);
|
|
if(!MySQL_CheckBan(ip,0,false,0)) {
|
|
msg.id = id;
|
|
msg.type = type;
|
|
msg.room = 0;
|
|
logPrintf(mysqlfile, "Executing MySQL Query: %s\n", servquery);
|
|
if(mysql_query(conn, servquery)) {
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
} else {
|
|
res = mysql_store_result(conn);
|
|
logPrintf(logfile, "Found %d servers...\n", mysql_num_rows(res));
|
|
while ((row = mysql_fetch_row(res)) != NULL)
|
|
{
|
|
if(strcmp(row[4], "0") == 1)
|
|
snprintf(str, sizeof str, "IP\t\t: %s\nPort\t\t: %s\nHostname\t: %s\nVersion\t\t: %s\nPermanent\t: %s\n", row[0], row[1], row[2], row[3], "Yes");
|
|
else
|
|
snprintf(str, sizeof str, "IP\t\t: %s\nPort\t\t: %s\nHostname\t: %s\nVersion\t\t: %s\nPermanent\t: %s\n", row[0], row[1], row[2], row[3], "No");
|
|
|
|
msg.length = (INT32)(strlen(str)+1); // send also the '\0'
|
|
strcpy(msg.buffer, str);
|
|
dbgPrintf(CYAN, "Writing: (%d)\n%s\n", msg.length, msg.buffer);
|
|
writecode = server_socket.write(&msg);
|
|
if (writecode < 0)
|
|
{
|
|
dbgPrintf(LIGHTRED, "Write error... %d client %d deleted\n", writecode, id);
|
|
return;
|
|
}
|
|
}
|
|
mysql_free_result(res);
|
|
}
|
|
} else {
|
|
logPrintf(logfile, "IP %s is banned so do nothing! Let them find out for themselves.\n", ip);
|
|
}
|
|
}
|
|
|
|
void MySQL_RemoveServer(char *ip, char *port, char *name, char *version) {
|
|
char escapedName[255];
|
|
char updatequery[5000];
|
|
char updatequeryp1[5000] = "UPDATE `ms_servers` SET upnow = '0' WHERE `ip` = '%s' AND `port` = '%s' AND `permanent` = '0'";
|
|
MySQL_Conn(false);
|
|
mysql_real_escape_string(conn, escapedName, name, (unsigned long)strlen(name));
|
|
sprintf(updatequery, updatequeryp1, ip, port);
|
|
logPrintf(mysqlfile, "Executing MySQL Query: %s\n", updatequery);
|
|
if(mysql_query(conn, updatequery)) {
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
} else {
|
|
logPrintf(logfile, "Server: %s:%s || Name: %s || Version: %s removed successfully.\n", ip, port, name, version);
|
|
}
|
|
}
|
|
|
|
/* LIVE FUNCTIONS */
|
|
void LIVE_AuthUser(UINT32 id, char *buffer)
|
|
{
|
|
msg_live_auth_t *auth;
|
|
char escapedName[255];
|
|
char saltedPassword[62];
|
|
char hashedPassword[34];
|
|
char namequery[1000];
|
|
char namequeryp1[1000] = "SELECT username,userid,salt,password FROM `user` WHERE `username` = '%s' LIMIT 1";
|
|
char userquery[1000];
|
|
char userqueryp1[1000] = "SELECT userid,username,live_authkey,live_publickey FROM `user` WHERE `userid` = '%s' LIMIT 1";
|
|
msg_t msg2;
|
|
char ip[32];
|
|
int writecode;
|
|
msg2.type = LIVE_INVALID_USER;
|
|
|
|
auth = (msg_live_auth_t *)buffer;
|
|
|
|
logPrintf(logfile, "Got User %s Pass %s\n", auth->username, auth->password);
|
|
// retrieve the true ip of the server
|
|
strcpy(ip, server_socket.getClientIP(id));
|
|
MySQL_Conn(false);
|
|
msg2.id = id;
|
|
logPrintf(logfile, "Check their ban status first... (ID %d)\n",id);
|
|
if(MySQL_CheckBan(ip,0,false,0))
|
|
return;
|
|
mysql_real_escape_string(conn, escapedName, auth->username, (unsigned long)strlen(auth->username));
|
|
sprintf(namequery, namequeryp1, escapedName);
|
|
logPrintf(mysqlfile, "Executing MySQL Query: %s\n", namequery);
|
|
if(mysql_query(conn, namequery)) {
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
} else {
|
|
res = mysql_store_result(conn);
|
|
if(mysql_num_rows(res) < 1)
|
|
{
|
|
logPrintf(logfile, "Invalid user: %s\n", auth->username);
|
|
msg2.type = LIVE_INVALID_USER;
|
|
}
|
|
else
|
|
{
|
|
row = mysql_fetch_row(res);
|
|
sprintf(saltedPassword, "%s%s", auth->password, row[2]);
|
|
logPrintf(logfile, "Pre-hash: %s (Salt is %s)\n", saltedPassword, row[2]);
|
|
sprintf(hashedPassword, md5(saltedPassword).c_str());
|
|
logPrintf(logfile, "Got Hashed Password: %s\n", hashedPassword);
|
|
if(strcmp(hashedPassword, row[3]) != 0)
|
|
{
|
|
msg2.type = LIVE_INVALID_USER;
|
|
logPrintf(logfile, "INVALID USER!\n");
|
|
} else {
|
|
msg2.type = LIVE_SEND_USER;
|
|
sprintf(userquery, userqueryp1, row[1]);
|
|
mysql_free_result(res);
|
|
logPrintf(mysqlfile, "Executing MySQL Query: %s\n", userquery);
|
|
if(mysql_query(conn, userquery)) {
|
|
logPrintf(errorfile, "MYSQL ERROR: %s\n", mysql_error(conn));
|
|
MySQL_Conn(true);
|
|
} else {
|
|
res = mysql_store_result(conn);
|
|
if(mysql_num_rows(res) < 1)
|
|
{
|
|
logPrintf(errorfile, "User Information Error, invalid USER ID!\n");
|
|
msg2.type = LIVE_INVALID_USER;
|
|
} else {
|
|
row = mysql_fetch_row(res);
|
|
msg_live_user_t *user = (msg_live_user_t *) msg2.buffer;
|
|
|
|
user->header[0] = '\0'; // nothing interresting in it (for now)
|
|
user->uid = atoi(row[0]);
|
|
strcpy(user->username, row[1]);
|
|
logPrintf(logfile, "Found user: %s, id %s\n", row[1], row[0]);
|
|
logPrintf(logfile, "Sending User %s with ID %d\n", user->username, user->uid);
|
|
|
|
msg2.length = sizeof (msg_live_user_t);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
msg2.length = sizeof msg2.buffer;
|
|
msg2.room = 0;
|
|
logPrintf(logfile, "Sending message?\n");
|
|
writecode = server_socket.write(&msg2);
|
|
logPrintf(logfile, "Message sent! :D\n");
|
|
if (writecode < 0)
|
|
{
|
|
dbgPrintf(LIGHTRED, "Write error... %d client %d "
|
|
"deleted\n", writecode, id);
|
|
return;
|
|
}
|
|
mysql_free_result(res);
|
|
}
|
|
|
|
/*
|
|
** sendServersInformations()
|
|
*/
|
|
static void sendServersInformations(UINT32 id, UINT32 room)
|
|
{
|
|
msg_t msg;
|
|
int writecode;
|
|
char ip[16];
|
|
strcpy(ip, server_socket.getClientIP(id));
|
|
(void)room;
|
|
logPrintf(logfile, "Sending servers informations\n");
|
|
msg.id = id;
|
|
msg.type = SEND_SERVER_MSG;
|
|
msg.room = 0;
|
|
MySQL_ListServServers(id, SEND_SERVER_MSG, ip); // Awesome new MySQL Code!
|
|
msg.length = 0;
|
|
//dbgPrintf(CYAN, "Writing: (%d) %s\n", msg.length, "");
|
|
writecode = server_socket.write(&msg);
|
|
if (writecode < 0)
|
|
{
|
|
dbgPrintf(LIGHTRED, "Write error... %d client %d deleted\n",
|
|
writecode, id);
|
|
}
|
|
}
|
|
|
|
/*
|
|
** sendShortServersInformations()
|
|
*/
|
|
static void sendShortServersInformations(UINT32 id, UINT32 room)
|
|
{
|
|
msg_t msg;
|
|
int writecode;
|
|
char ip[16];
|
|
strcpy(ip, server_socket.getClientIP(id));
|
|
|
|
logPrintf(logfile, "Sending short servers informations\n");
|
|
msg.id = id;
|
|
msg.type = SEND_SHORT_SERVER_MSG;
|
|
MySQL_ListServers(id, SEND_SHORT_SERVER_MSG, ip, room); // New code that's full of win!
|
|
logPrintf(logfile, "Packet Room: %d\n", room);
|
|
msg.length = 0;
|
|
writecode = server_socket.write(&msg);
|
|
if (writecode < 0)
|
|
{
|
|
dbgPrintf(LIGHTRED, "Write error... %d client %d deleted\n",
|
|
writecode, id);
|
|
}
|
|
}
|
|
|
|
/*
|
|
** sendRoomInformations()
|
|
*/
|
|
static void sendRoomInformations(UINT32 id, UINT32 room, bool type)
|
|
{
|
|
msg_t msg;
|
|
int writecode;
|
|
char ip[16];
|
|
strcpy(ip, server_socket.getClientIP(id));
|
|
(void)room;
|
|
logPrintf(logfile, "Sending room information\n");
|
|
msg.id = id;
|
|
msg.type = SEND_ROOMS_MSG;
|
|
msg.room = 0;
|
|
MySQL_ListRooms(id, SEND_ROOMS_MSG, ip, type); // New code that's full of win!
|
|
msg.length = 0;
|
|
writecode = server_socket.write(&msg);
|
|
if (writecode < 0)
|
|
{
|
|
dbgPrintf(LIGHTRED, "Write error... %d client %d deleted\n",
|
|
writecode, id);
|
|
}
|
|
}
|
|
|
|
/*
|
|
** sendVersionInformations()
|
|
*/
|
|
static void sendVersionInformations(UINT32 id, UINT32 modid, char *modversion)
|
|
{
|
|
msg_t msg;
|
|
int writecode;
|
|
char ip[16];
|
|
strcpy(ip, server_socket.getClientIP(id));
|
|
|
|
logPrintf(logfile, "Sending version information\n");
|
|
msg.id = id;
|
|
msg.type = SEND_VERSION_MSG;
|
|
msg.room = 0;
|
|
MySQL_CheckVersion(id, SEND_VERSION_MSG, ip, modid, modversion); // New code that's full of win!
|
|
msg.length = sizeof msg.buffer;
|
|
writecode = server_socket.write(&msg);
|
|
if (writecode < 0)
|
|
{
|
|
dbgPrintf(LIGHTRED, "Write error... %d client %d deleted\n",
|
|
writecode, id);
|
|
}
|
|
}
|
|
|
|
/*
|
|
** addServer()
|
|
*/
|
|
static void addServer(int id, char *buffer, bool firstadd)
|
|
{
|
|
msg_server_t *info;
|
|
char oldversion = 0;
|
|
|
|
//TODO: Be sure there is no flood from a given IP:
|
|
// If a host need more than 2 servers, then it should be registrated
|
|
// manually
|
|
|
|
info = (msg_server_t *)buffer;
|
|
|
|
// I want to be sure the informations are correct, of course!
|
|
info->port[sizeof (info->port)-1] = '\0';
|
|
info->name[sizeof (info->name)-1] = '\0';
|
|
info->version[sizeof (info->version)-1] = '\0';
|
|
|
|
logPrintf(logfile, "addServer(): Version = \"%s\"\n", info->version);
|
|
logPrintf(logfile, "addServer(): Key = \"%s\"\n", info->key);
|
|
|
|
// retrieve the true ip of the server
|
|
strcpy(info->ip, server_socket.getClientIP(id));
|
|
//strcpy(info->port, server_socket.getClientPort(id));
|
|
|
|
if (info->version[0] == '1' &&
|
|
info->version[1] == '.' &&
|
|
info->version[2] == '0' &&
|
|
info->version[3] == '9' &&
|
|
info->version[4] == '.')
|
|
{
|
|
if ((info->version[5] == '2') || (info->version[5] == '3'))
|
|
{
|
|
oldversion = 1;
|
|
}
|
|
}
|
|
|
|
if (info->version[0] == '1' &&
|
|
info->version[1] == '.' &&
|
|
info->version[2] == '6' &&
|
|
info->version[3] == '9' &&
|
|
info->version[4] == '.' &&
|
|
info->version[5] == '6')
|
|
{
|
|
oldversion = 1;
|
|
}
|
|
|
|
if (!oldversion)
|
|
{
|
|
MySQL_AddServer(info->ip, info->port, info->name, info->version, info->room, firstadd, info->key);
|
|
}
|
|
else
|
|
{
|
|
logPrintf(logfile, "Not adding the temporary server: %s %s %s %s\n", info->ip, info->port, info->name, info->version);
|
|
}
|
|
}
|
|
|
|
/*
|
|
** removeServer()
|
|
*/
|
|
static void removeServer(int id, char *buffer)
|
|
{
|
|
msg_server_t *info;
|
|
|
|
info = (msg_server_t *)buffer;
|
|
|
|
// I want to be sure the informations are correct, of course!
|
|
info->port[sizeof (info->port)-1] = '\0';
|
|
info->name[sizeof (info->name)-1] = '\0';
|
|
info->version[sizeof (info->version)-1] = '\0';
|
|
|
|
// retrieve the true ip of the server
|
|
strcpy(info->ip, server_socket.getClientIP(id));
|
|
|
|
logPrintf(logfile, "Removing the temporary server: %s %s %s %s\n", info->ip, info->port, info->name, info->version);
|
|
MySQL_RemoveServer(info->ip, info->port, info->name, info->version); // New and win.
|
|
}
|
|
|
|
/*
|
|
** analyseUDP()
|
|
*/
|
|
static int analyseUDP(size_t size, const char *buffer)
|
|
{
|
|
// this would be the part of reading the PT_SERVERINFO packet,
|
|
// but i'm not about to backport that sloppy code
|
|
(void)size;
|
|
(void)buffer;
|
|
return INVALID_MSG;
|
|
}
|
|
|
|
/*
|
|
** UDPMessage()
|
|
*/
|
|
static int UDPMessage(size_t size, const char *buffer)
|
|
{
|
|
if (size == 4)
|
|
{
|
|
return 0;
|
|
}
|
|
else if (size == sizeof(ms_holepunch_packet_t))
|
|
{
|
|
return 0;
|
|
}
|
|
else if (size > 58)
|
|
{
|
|
dbgPrintf(LIGHTGREEN, "Got a UDP %d byte long message\n", size);
|
|
return analyseUDP(size, buffer);
|
|
}
|
|
else
|
|
return INVALID_MSG;
|
|
}
|
|
|
|
/*
|
|
** analyseMessage()
|
|
*/
|
|
static int analyseMessage(msg_t *msg)
|
|
{
|
|
switch (msg->type)
|
|
{
|
|
case UDP_RECV_MSG:
|
|
return UDPMessage(msg->length, msg->buffer);
|
|
case ACCEPT_MSG:
|
|
break;
|
|
case ADD_SERVER_MSG:
|
|
addServer(msg->id, msg->buffer, true);
|
|
break;
|
|
case PING_SERVER_MSG:
|
|
addServer(msg->id, msg->buffer, false);
|
|
break;
|
|
case REMOVE_SERVER_MSG:
|
|
removeServer(msg->id, msg->buffer);
|
|
break;
|
|
case GET_SERVER_MSG:
|
|
sendServersInformations(msg->id, msg->room);
|
|
break;
|
|
case GET_SHORT_SERVER_MSG:
|
|
sendShortServersInformations(msg->id, msg->room);
|
|
break;
|
|
case GET_ROOMS_MSG:
|
|
sendRoomInformations(msg->id, msg->room, 0);
|
|
break;
|
|
case GET_ROOMS_HOST_MSG:
|
|
sendRoomInformations(msg->id, msg->room, 1);
|
|
break;
|
|
case GET_VERSION_MSG:
|
|
sendVersionInformations(msg->id, msg->room, msg->buffer);
|
|
break;
|
|
case LIVE_AUTH_USER:
|
|
LIVE_AuthUser(msg->id, msg->buffer);
|
|
break;
|
|
default:
|
|
return INVALID_MSG;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
** main()
|
|
*/
|
|
int main(int argc, char *argv[])
|
|
{
|
|
msg_t msg;
|
|
#ifdef __unix__
|
|
pid_t pid;
|
|
char pidstr[12];
|
|
#endif
|
|
|
|
if (argc <= 1)
|
|
{
|
|
fprintf(stderr, "usage: %s port\n", argv[0]);
|
|
exit(1);
|
|
}
|
|
|
|
if (server_socket.listen(argv[1]) < 0)
|
|
{
|
|
fprintf(stderr, "Error while initializing the server; port being used! Try killing the other Master Server.\n");
|
|
exit(2);
|
|
}
|
|
|
|
logfile = openFile("server.log");
|
|
errorfile = openFile("error.log");
|
|
mysqlfile = openFile("mysql.log");
|
|
sockfile = openFile("sockets.log");
|
|
MySQL_Conn(false);
|
|
#if !defined (DEBUG) && defined (__unix__)
|
|
pid = fork();
|
|
switch (pid)
|
|
{
|
|
case 0: break; // child
|
|
case -1: printf("Error while launching the server in background\n"); return -1;
|
|
default:
|
|
pidfile = fopen("msd.pid", "w+");
|
|
sprintf(pidstr, "%d", pid);
|
|
fputs(pidstr, pidfile);
|
|
fclose(pidfile);
|
|
return 0; // parent: keep child in background
|
|
}
|
|
#endif
|
|
srand((unsigned)time(NULL)); // Alam: GUIDs
|
|
for (;;)
|
|
{
|
|
memset(&msg, 0, sizeof (msg)); // remove previous message
|
|
if (!server_socket.read(&msg))
|
|
{
|
|
// valid message: header message seems ok
|
|
analyseMessage(&msg);
|
|
//servers_list.show(); // for debug purpose
|
|
}
|
|
}
|
|
|
|
#ifndef _MSC_VER
|
|
/* NOTREACHED */
|
|
return 0;
|
|
#endif
|
|
}
|