343 lines
12 KiB
C++
343 lines
12 KiB
C++
//////////////////////////////////////////////////////////////////////////////////////
|
|
// akashi - a server for Attorney Online 2 //
|
|
// Copyright (C) 2020 scatterflower //
|
|
// //
|
|
// This program is free software: you can redistribute it and/or modify //
|
|
// it under the terms of the GNU Affero General Public License as //
|
|
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details. //
|
|
// //
|
|
// You should have received a copy of the GNU Affero General Public License //
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>. //
|
|
//////////////////////////////////////////////////////////////////////////////////////
|
|
#include "include/server.h"
|
|
|
|
Server::Server(int p_port, int p_ws_port, QObject* parent) :
|
|
QObject(parent),
|
|
player_count(0),
|
|
port(p_port),
|
|
ws_port(p_ws_port)
|
|
{
|
|
server = new QTcpServer(this);
|
|
connect(server, SIGNAL(newConnection()), this, SLOT(clientConnected()));
|
|
|
|
timer = new QTimer();
|
|
|
|
db_manager = new DBManager();
|
|
|
|
}
|
|
|
|
void Server::start()
|
|
{
|
|
QSettings config("config/config.ini", QSettings::IniFormat);
|
|
config.beginGroup("Options");
|
|
QString bind_ip = config.value("ip", "all").toString();
|
|
QHostAddress bind_addr;
|
|
if (bind_ip == "all")
|
|
bind_addr = QHostAddress::Any;
|
|
else
|
|
bind_addr = QHostAddress(bind_ip);
|
|
if (bind_addr.protocol() != QAbstractSocket::IPv4Protocol && bind_addr.protocol() != QAbstractSocket::IPv6Protocol && bind_addr != QHostAddress::Any) {
|
|
qDebug() << bind_ip << "is an invalid IP address to listen on! Server not starting, check your config.";
|
|
}
|
|
if (!server->listen(bind_addr, port)) {
|
|
qDebug() << "Server error:" << server->errorString();
|
|
}
|
|
else {
|
|
qDebug() << "Server listening on" << port;
|
|
}
|
|
|
|
discord = new Discord(this);
|
|
handleDiscordIntegration();
|
|
|
|
if (ConfigManager::advertiseHTTPServer()) {
|
|
httpAdvertiserTimer = new QTimer(this);
|
|
httpAdvertiser = new HTTPAdvertiser();
|
|
|
|
connect(httpAdvertiserTimer, &QTimer::timeout,
|
|
httpAdvertiser, &HTTPAdvertiser::msAdvertiseServer);
|
|
connect(this, &Server::setHTTPConfiguration,
|
|
httpAdvertiser, &HTTPAdvertiser::setAdvertiserSettings);
|
|
connect(this, &Server::updateHTTPConfiguration,
|
|
httpAdvertiser, &HTTPAdvertiser::updateAdvertiserSettings);
|
|
setHTTPAdvertiserConfig();
|
|
httpAdvertiserTimer->start(300000);
|
|
}
|
|
|
|
logger = new ULogger(this);
|
|
connect(this, &Server::logConnectionAttempt,
|
|
logger, &ULogger::logConnectionAttempt);
|
|
|
|
proxy = new WSProxy(port, ws_port, this);
|
|
if(ws_port != -1)
|
|
proxy->start();
|
|
|
|
QFile char_list("config/characters.txt");
|
|
char_list.open(QIODevice::ReadOnly | QIODevice::Text);
|
|
while (!char_list.atEnd()) {
|
|
characters.append(char_list.readLine().trimmed());
|
|
}
|
|
char_list.close();
|
|
|
|
QFile music_file("config/music.txt");
|
|
music_file.open(QIODevice::ReadOnly | QIODevice::Text);
|
|
while (!music_file.atEnd()) {
|
|
music_list.append(music_file.readLine().trimmed());
|
|
}
|
|
music_file.close();
|
|
if(music_list[0].contains(".")) // Add a default category if none exists
|
|
music_list.insert(0, "==Music==");
|
|
|
|
QFile bg_file("config/backgrounds.txt");
|
|
bg_file.open(QIODevice::ReadOnly | QIODevice::Text);
|
|
while (!bg_file.atEnd()) {
|
|
backgrounds.append(bg_file.readLine().trimmed());
|
|
}
|
|
bg_file.close();
|
|
|
|
QSettings areas_ini("config/areas.ini", QSettings::IniFormat);
|
|
area_names = areas_ini.childGroups(); // invisibly does a lexicographical sort, because Qt is great like that
|
|
std::sort(area_names.begin(), area_names.end(), [] (const QString &a, const QString &b) {return a.split(":")[0].toInt() < b.split(":")[0].toInt();});
|
|
QStringList sanitized_area_names;
|
|
QStringList raw_area_names = area_names;
|
|
for (const QString &area_name : qAsConst(area_names)) {
|
|
QStringList name_split = area_name.split(":");
|
|
name_split.removeFirst();
|
|
QString area_name_sanitized = name_split.join(":");
|
|
sanitized_area_names.append(area_name_sanitized);
|
|
}
|
|
area_names = sanitized_area_names;
|
|
for (int i = 0; i < raw_area_names.length(); i++) {
|
|
QString area_name = raw_area_names[i];
|
|
areas.insert(i, new AreaData(area_name, i));
|
|
}
|
|
connect(&next_message_timer, SIGNAL(timeout()), this, SLOT(allowMessage()));
|
|
}
|
|
|
|
void Server::clientConnected()
|
|
{
|
|
QTcpSocket* socket = server->nextPendingConnection();
|
|
int user_id;
|
|
QList<int> user_ids;
|
|
for (AOClient* client : qAsConst(clients)) {
|
|
user_ids.append(client->id);
|
|
}
|
|
for (user_id = 0; user_id <= player_count; user_id++) {
|
|
if (user_ids.contains(user_id))
|
|
continue;
|
|
else
|
|
break;
|
|
}
|
|
AOClient* client = new AOClient(this, socket, this, user_id);
|
|
|
|
int multiclient_count = 1;
|
|
bool is_at_multiclient_limit = false;
|
|
client->calculateIpid();
|
|
auto ban = db_manager->isIPBanned(client->getIpid());
|
|
bool is_banned = ban.first;
|
|
for (AOClient* joined_client : qAsConst(clients)) {
|
|
if (client->remote_ip.isEqual(joined_client->remote_ip))
|
|
multiclient_count++;
|
|
}
|
|
|
|
if (multiclient_count > ConfigManager::multiClientLimit() && !client->remote_ip.isLoopback()) // TODO: make this configurable
|
|
is_at_multiclient_limit = true;
|
|
|
|
if (is_banned) {
|
|
QString reason = ban.second;
|
|
AOPacket ban_reason("BD", {reason});
|
|
socket->write(ban_reason.toUtf8());
|
|
}
|
|
if (is_banned || is_at_multiclient_limit) {
|
|
socket->flush();
|
|
client->deleteLater();
|
|
socket->close();
|
|
return;
|
|
}
|
|
|
|
clients.append(client);
|
|
connect(socket, &QTcpSocket::disconnected, client,
|
|
&AOClient::clientDisconnected);
|
|
connect(socket, &QTcpSocket::disconnected, this, [=] {
|
|
clients.removeAll(client);
|
|
client->deleteLater();
|
|
});
|
|
connect(socket, &QTcpSocket::readyRead, client, &AOClient::clientData);
|
|
|
|
AOPacket decryptor("decryptor", {"NOENCRYPT"}); // This is the infamous workaround for
|
|
// tsuserver4. It should disable fantacrypt
|
|
// completely in any client 2.4.3 or newer
|
|
client->sendPacket(decryptor);
|
|
hookupLogger(client);
|
|
#ifdef NET_DEBUG
|
|
qDebug() << client->remote_ip.toString() << "connected";
|
|
#endif
|
|
}
|
|
|
|
void Server::updateCharsTaken(AreaData* area)
|
|
{
|
|
QStringList chars_taken;
|
|
for (const QString &cur_char : qAsConst(characters)) {
|
|
chars_taken.append(area->charactersTaken().contains(getCharID(cur_char))
|
|
? QStringLiteral("-1")
|
|
: QStringLiteral("0"));
|
|
}
|
|
|
|
AOPacket response_cc("CharsCheck", chars_taken);
|
|
|
|
for (AOClient* client : qAsConst(clients)) {
|
|
if (client->current_area == area->index()){
|
|
if (!client->is_charcursed)
|
|
client->sendPacket(response_cc);
|
|
else {
|
|
QStringList chars_taken_cursed = getCursedCharsTaken(client, chars_taken);
|
|
AOPacket response_cc_cursed("CharsCheck", chars_taken_cursed);
|
|
client->sendPacket(response_cc_cursed);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
QStringList Server::getCursedCharsTaken(AOClient* client, QStringList chars_taken)
|
|
{
|
|
QStringList chars_taken_cursed;
|
|
for (int i = 0; i < chars_taken.length(); i++) {
|
|
if (!client->charcurse_list.contains(i))
|
|
chars_taken_cursed.append("-1");
|
|
else
|
|
chars_taken_cursed.append(chars_taken.value(i));
|
|
}
|
|
return chars_taken_cursed;
|
|
}
|
|
|
|
void Server::broadcast(AOPacket packet, int area_index)
|
|
{
|
|
for (AOClient* client : qAsConst(clients)) {
|
|
if (client->current_area == area_index)
|
|
client->sendPacket(packet);
|
|
}
|
|
}
|
|
|
|
void Server::broadcast(AOPacket packet)
|
|
{
|
|
for (AOClient* client : qAsConst(clients)) {
|
|
client->sendPacket(packet);
|
|
}
|
|
}
|
|
|
|
QList<AOClient*> Server::getClientsByIpid(QString ipid)
|
|
{
|
|
QList<AOClient*> return_clients;
|
|
for (AOClient* client : qAsConst(clients)) {
|
|
if (client->getIpid() == ipid)
|
|
return_clients.append(client);
|
|
}
|
|
return return_clients;
|
|
}
|
|
|
|
AOClient* Server::getClientByID(int id)
|
|
{
|
|
for (AOClient* client : qAsConst(clients)) {
|
|
if (client->id == id)
|
|
return client;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
int Server::getCharID(QString char_name)
|
|
{
|
|
for (const QString &character : qAsConst(characters)) {
|
|
if (character.toLower() == char_name.toLower()) {
|
|
return characters.indexOf(QRegExp(character, Qt::CaseInsensitive));
|
|
}
|
|
}
|
|
return -1; // character does not exist
|
|
}
|
|
|
|
void Server::setHTTPAdvertiserConfig()
|
|
{
|
|
advertiser_config config;
|
|
config.name = ConfigManager::serverName();
|
|
config.description = ConfigManager::serverDescription();
|
|
config.port = ConfigManager::serverPort();
|
|
config.ws_port = ConfigManager::webaoPort();
|
|
config.players = ConfigManager::maxPlayers();
|
|
config.masterserver = ConfigManager::advertiserHTTPIP();
|
|
config.debug = ConfigManager::advertiserHTTPDebug();
|
|
emit setHTTPConfiguration(config);
|
|
}
|
|
|
|
void Server::updateHTTPAdvertiserConfig()
|
|
{
|
|
update_advertiser_config config;
|
|
config.name = ConfigManager::serverName();
|
|
config.description = ConfigManager::serverDescription();
|
|
config.players = ConfigManager::maxPlayers();
|
|
config.masterserver = ConfigManager::advertiserHTTPIP();
|
|
config.debug = ConfigManager::advertiserHTTPDebug();
|
|
emit updateHTTPConfiguration(config);
|
|
|
|
}
|
|
|
|
void Server::allowMessage()
|
|
{
|
|
can_send_ic_messages = true;
|
|
}
|
|
|
|
void Server::handleDiscordIntegration()
|
|
{
|
|
// Prevent double connecting by preemtively disconnecting them.
|
|
disconnect(this, nullptr, discord, nullptr);
|
|
|
|
if (ConfigManager::discordWebhookEnabled()) {
|
|
if (ConfigManager::discordModcallWebhookEnabled())
|
|
connect(this, &Server::modcallWebhookRequest,
|
|
discord, &Discord::onModcallWebhookRequested);
|
|
|
|
if (ConfigManager::discordBanWebhookEnabled())
|
|
connect(this, &Server::banWebhookRequest,
|
|
discord, &Discord::onBanWebhookRequested);
|
|
|
|
if (ConfigManager::discordUptimeEnabled())
|
|
discord->startUptimeTimer();
|
|
else
|
|
discord->stopUptimeTimer();
|
|
}
|
|
return;
|
|
}
|
|
|
|
void Server::hookupLogger(AOClient* client)
|
|
{
|
|
connect(client, &AOClient::logIC,
|
|
logger, &ULogger::logIC);
|
|
connect(client, &AOClient::logOOC,
|
|
logger, &ULogger::logOOC);
|
|
connect(client, &AOClient::logLogin,
|
|
logger, &ULogger::logLogin);
|
|
connect(client, &AOClient::logCMD,
|
|
logger, &ULogger::logCMD);
|
|
connect(client, &AOClient::logBan,
|
|
logger, &ULogger::logBan);
|
|
connect(client, &AOClient::logKick,
|
|
logger, &ULogger::logKick);
|
|
connect(client, &AOClient::logModcall,
|
|
logger, &ULogger::logModcall);
|
|
}
|
|
|
|
Server::~Server()
|
|
{
|
|
for (AOClient* client : qAsConst(clients)) {
|
|
client->deleteLater();
|
|
}
|
|
server->deleteLater();
|
|
proxy->deleteLater();
|
|
discord->deleteLater();
|
|
|
|
delete db_manager;
|
|
}
|