////////////////////////////////////////////////////////////////////////////////////// // 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 . // ////////////////////////////////////////////////////////////////////////////////////// #include "include/server.h" Server::Server(int p_port, int p_ws_port, QObject* parent) : QObject(parent), m_player_count(0), port(p_port), ws_port(p_ws_port) { server = new QTcpServer(this); connect(server, SIGNAL(newConnection()), this, SLOT(clientConnected())); proxy = new WSProxy(port, ws_port, this); if(ws_port != -1) proxy->start(); timer = new QTimer(); db_manager = new DBManager(); //We create it, even if its not used later on. discord = new Discord(this); logger = new ULogger(this); connect(this, &Server::logConnectionAttempt, logger, &ULogger::logConnectionAttempt); } void Server::start() { QString bind_ip = ConfigManager::bindIP(); 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; } //Checks if any Discord webhooks are enabled. handleDiscordIntegration(); //Construct modern advertiser if enabled in config if (ConfigManager::advertiseServer()) { AdvertiserTimer = new QTimer(this); ms3_Advertiser = new Advertiser(); connect(AdvertiserTimer, &QTimer::timeout, ms3_Advertiser, &Advertiser::msAdvertiseServer); connect(this, &Server::updatePlayerCount, ms3_Advertiser, &Advertiser::updatePlayerCount); connect(this, &Server::updateHTTPConfiguration, ms3_Advertiser, &Advertiser::updateAdvertiserSettings); emit updatePlayerCount(m_player_count); ms3_Advertiser->msAdvertiseServer(); AdvertiserTimer->start(300000); } //Get characters from config file m_characters = ConfigManager::charlist(); //Get musiclist from config file m_music_list = ConfigManager::musiclist(); //Get backgrounds from config file m_backgrounds = ConfigManager::backgrounds(); //Assembles the area list m_area_names = ConfigManager::sanitizedAreaNames(); QStringList raw_area_names = ConfigManager::rawAreaNames(); for (int i = 0; i < raw_area_names.length(); i++) { QString area_name = raw_area_names[i]; AreaData* l_area = new AreaData(area_name, i); m_areas.insert(i, l_area); connect(l_area, &AreaData::playJukeboxSong, this, QOverload::of(&Server::broadcast)); } //Loads the command help information. This is not stored inside the server. ConfigManager::loadCommandHelp(); //Get IP bans m_ipban_list = ConfigManager::iprangeBans(); //Rate-Limiter for IC-Chat connect(&next_message_timer, &QTimer::timeout, this, &Server::allowMessage); //Prepare player IDs and reference hash. for (int i = 0; i <= ConfigManager::maxPlayers() - 1; i++){ m_available_ids.enqueue(i); m_clients_ids.insert(i, nullptr); } } void Server::clientConnected() { QTcpSocket* socket = server->nextPendingConnection(); //Too many players. Reject connection! //This also enforces the maximum playercount. if (m_available_ids.empty()) { AOPacket disconnect_reason("BD", {"Maximum playercount has been reached."}); socket->write(disconnect_reason.toUtf8()); socket->flush(); socket->close(); socket->deleteLater(); return; } int user_id = m_available_ids.dequeue(); AOClient* client = new AOClient(this, socket, this, user_id); m_clients_ids.insert(user_id, client); 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(m_clients)) { if (client->m_remote_ip.isEqual(joined_client->m_remote_ip)) multiclient_count++; } if (multiclient_count > ConfigManager::multiClientLimit() && !client->m_remote_ip.isLoopback()) 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(); markIDFree(user_id); return; } QHostAddress l_remote_ip = client->m_remote_ip; if (l_remote_ip.protocol() == QAbstractSocket::IPv6Protocol) { l_remote_ip = parseToIPv4(l_remote_ip); } if (isIPBanned(l_remote_ip)){ QString l_reason = "Your IP has been banned by a moderator."; AOPacket l_ban_reason("BD", {l_reason}); socket->write(l_ban_reason.toUtf8()); client->deleteLater(); socket->close(); markIDFree(user_id); return; } m_clients.append(client); connect(socket, &QTcpSocket::disconnected, client, &AOClient::clientDisconnected); connect(socket, &QTcpSocket::disconnected, this, [=] { m_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); hookupAOClient(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(m_characters)) { chars_taken.append(area->charactersTaken().contains(getCharID(cur_char)) ? QStringLiteral("-1") : QStringLiteral("0")); } AOPacket response_cc("CharsCheck", chars_taken); for (AOClient* client : qAsConst(m_clients)) { if (client->m_current_area == area->index()){ if (!client->m_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->m_charcurse_list.contains(i)) chars_taken_cursed.append("-1"); else chars_taken_cursed.append(chars_taken.value(i)); } return chars_taken_cursed; } QHostAddress Server::parseToIPv4(QHostAddress f_remote_ip) { bool l_ok; QHostAddress l_remote_ip = f_remote_ip; QHostAddress l_temp_remote_ip = QHostAddress(f_remote_ip.toIPv4Address(&l_ok)); if (l_ok) { l_remote_ip = l_temp_remote_ip; } return l_remote_ip; } void Server::reloadSettings() { ConfigManager::reloadSettings(); emit reloadRequest(ConfigManager::serverName(), ConfigManager::serverDescription()); emit updateHTTPConfiguration(); handleDiscordIntegration(); logger->loadLogtext(); m_music_list = ConfigManager::musiclist(); m_ipban_list = ConfigManager::iprangeBans(); } void Server::broadcast(AOPacket packet, int area_index) { for (AOClient* client : qAsConst(m_clients)) { if (client->m_current_area == area_index) client->sendPacket(packet); } } void Server::broadcast(AOPacket packet) { for (AOClient* client : qAsConst(m_clients)) { client->sendPacket(packet); } } void Server::broadcast(AOPacket packet, TARGET_TYPE target) { for (AOClient* l_client : qAsConst(m_clients)) { switch (target) { case TARGET_TYPE::MODCHAT: if (l_client->checkAuth(l_client->ACLFlags.value("MODCHAT"))) { l_client->sendPacket(packet); } break; case TARGET_TYPE::ADVERT: if (l_client->m_advert_enabled) { l_client->sendPacket(packet); } break; default: break; } } } void Server::broadcast(AOPacket packet, AOPacket other_packet, TARGET_TYPE target) { switch (target) { case TARGET_TYPE::AUTHENTICATED: for (AOClient* client : qAsConst(m_clients)){ if (client->m_authenticated) { client->sendPacket(other_packet); } else { client->sendPacket(packet); } } default: //Unimplemented, so not handled. break; } } QList Server::getClientsByIpid(QString ipid) { QList return_clients; for (AOClient* client : qAsConst(m_clients)) { if (client->getIpid() == ipid) return_clients.append(client); } return return_clients; } AOClient* Server::getClientByID(int id) { return m_clients_ids.value(id); } int Server::getCharID(QString char_name) { for (const QString &character : qAsConst(m_characters)) { if (character.toLower() == char_name.toLower()) { return m_characters.indexOf(QRegExp(character, Qt::CaseInsensitive, QRegExp::FixedString)); } } return -1; // character does not exist } QQueue Server::getAreaBuffer(const QString &f_areaName) { return logger->buffer(f_areaName); } 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::markIDFree(const int &f_user_id) { m_available_ids.enqueue(f_user_id); m_clients_ids.insert(f_user_id, nullptr); } void Server::hookupAOClient(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); connect(client, &AOClient::clientSuccessfullyDisconnected, this, &Server::markIDFree); } bool Server::isIPBanned(QHostAddress f_remote_IP) { bool l_match_found = false; for(const QString &l_ipban : qAsConst(m_ipban_list)) { if (f_remote_IP.isInSubnet(QHostAddress::parseSubnet(l_ipban))) { l_match_found = true; break; } } return l_match_found; } Server::~Server() { for (AOClient* client : qAsConst(m_clients)) { client->deleteLater(); } server->deleteLater(); proxy->deleteLater(); discord->deleteLater(); delete db_manager; }