diff --git a/bin/config_sample/config.ini b/bin/config_sample/config.ini index c3c4621..6a21d4d 100644 --- a/bin/config_sample/config.ini +++ b/bin/config_sample/config.ini @@ -60,6 +60,18 @@ message_floodguard=250 ; The URL of the server's remote repository, sent to the client during their initial handshake. Used by WebAO users for custom content. asset_url= +[ModernAdvertiser] +; Options for the HTTP based Masterserver. + +; Whether or not the server should appear on the master server. +advertise=false + +; Wether the advertiser prints additional debug info +debug=false + +; The IP address of the master server. Unless something happens to the default one, you shouldn't change this. +ms_ip=https://ms3.oldmud0.workers.dev/servers + [Dice] ; The maximum number of sides dice can be rolled with. max_value=100 diff --git a/core/core.pro b/core/core.pro index ea3e4ec..26f72c2 100644 --- a/core/core.pro +++ b/core/core.pro @@ -45,7 +45,8 @@ SOURCES += \ src/server.cpp \ src/testimony_recorder.cpp \ src/ws_client.cpp \ - src/ws_proxy.cpp + src/ws_proxy.cpp \ + src/http_advertiser.cpp HEADERS += include/advertiser.h \ include/aoclient.h \ @@ -58,4 +59,5 @@ HEADERS += include/advertiser.h \ include/logger.h \ include/server.h \ include/ws_client.h \ - include/ws_proxy.h + include/ws_proxy.h \ + include/http_advertiser.h diff --git a/core/include/config_manager.h b/core/include/config_manager.h index 852e28b..c661ef3 100644 --- a/core/include/config_manager.h +++ b/core/include/config_manager.h @@ -302,6 +302,22 @@ class ConfigManager { */ static QStringList gimpList(); + /** + * @brief Returns if the HTTP advertiser is constructed or not. + */ + static bool advertiseHTTPServer(); + + /** + * @brief Returns if the HTTP advertiser prints debug info to console. + */ + static bool advertiserHTTPDebug(); + + /** + * @brief Returns the IP or URL of the masterserver. + */ + static QUrl advertiserHTTPIP(); + + /** * @brief Sets the server's authorization type. * diff --git a/core/include/http_advertiser.h b/core/include/http_advertiser.h new file mode 100644 index 0000000..7440ae2 --- /dev/null +++ b/core/include/http_advertiser.h @@ -0,0 +1,115 @@ +////////////////////////////////////////////////////////////////////////////////////// +// 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 . // +////////////////////////////////////////////////////////////////////////////////////// +#ifndef HTTP_ADVERTISER_H +#define HTTP_ADVERTISER_H + +#include +#include + +//Don't question this. It needs to be here for some reason. +struct advertiser_config { + QString name; + QString description; + int port; + int ws_port; + int players; + QUrl masterserver; + bool debug; +}; + +/** + * @brief Represents the advertiser of the server. Sends current server information to masterserver. + */ +class HTTPAdvertiser : public QObject +{ + Q_OBJECT +public: + /** + * @brief Constructor for the HTTP_Advertiser class. + */ + explicit HTTPAdvertiser(); + + + /** + * @brief Deconstructor for the HTTP_Advertiser class. Yes, that's it. Can't say more about it. + */ + ~HTTPAdvertiser(); + +public slots: + + /** + * @brief Establishes a connection with masterserver to register or update the listing on the masterserver. + */ + void msAdvertiseServer(); + + /** + * @brief Reads the information send as a reply for further error handling. + * @param reply Response data from the masterserver. Information contained is send to the console if debug is enabled. + */ + void msRequestFinished(QNetworkReply *reply); + + /** + * @brief Sets the values being advertised to masterserver. + * @param config Configuration struct for the advertiser. Always includes ALL settings. + */ + void setAdvertiserSettings(advertiser_config config); + +private: + + /** + * @brief Pointer to the network manager, necessary to execute POST requests to the masterserver. + */ + QNetworkAccessManager* m_manager; + + /** + * @brief Name of the server send to the masterserver. Changing this will change the display name in the serverlist + */ + QString m_name; + + /** + * @brief Description of the server that is displayed in the client when the server is selected. + */ + QString m_description; + + /** + * @brief Client port for the AO2-Client. + */ + int m_port; + + /** + * @brief Websocket proxy port for WebAO users. + */ + int m_ws_port; + + /** + * @brief Maximum amount of clients that can be connected to the server. + */ + int m_players; + + /** + * @brief URL of the masterserver that m_manager posts to. This is almost never changed. + */ + QUrl m_masterserver; + + /** + * @brief Controls if network replies are printed to console. Should only be true if issues communicating with masterserver appear. + */ + bool m_debug; +}; + +#endif // HTTP_ADVERTISER_H diff --git a/core/include/server.h b/core/include/server.h index 05e1f48..3456b64 100644 --- a/core/include/server.h +++ b/core/include/server.h @@ -25,6 +25,7 @@ #include "include/db_manager.h" #include "include/discord.h" #include "include/config_manager.h" +#include "include/http_advertiser.h" #include #include @@ -137,6 +138,11 @@ class Server : public QObject { */ int getCharID(QString char_name); + /** + * @brief Creates an HTTP advertiser config struct and emits it using server::reloadHTTPRequest. + */ + void reloadHTTPAdvertiserConfig(); + /** * @brief The collection of all currently connected clients. */ @@ -227,6 +233,12 @@ class Server : public QObject { */ void reloadRequest(QString p_name, QString p_desc); + /** + * @brief Sends all necessary info for the new advertiser. + * @param Struct that contains all configuration for the advertiser + */ + void reloadHTTPRequest(struct advertiser_config config); + /** * @brief Sends a modcall webhook request, emitted by AOClient::pktModcall. * @@ -255,6 +267,16 @@ class Server : public QObject { */ Discord* discord; + /** + * @brief Handles HTTP server advertising. + */ + HTTPAdvertiser* httpAdvertiser; + + /** + * @brief Advertises the server in a regular intervall. + */ + QTimer* httpAdvertiserTimer; + /** * @brief The port through which the server will accept TCP connections. */ diff --git a/core/src/commands/moderation.cpp b/core/src/commands/moderation.cpp index 1582151..e7e4d46 100644 --- a/core/src/commands/moderation.cpp +++ b/core/src/commands/moderation.cpp @@ -412,6 +412,7 @@ void AOClient::cmdReload(int argc, QStringList argv) { ConfigManager::reloadSettings(); emit server->reloadRequest(ConfigManager::serverName(), ConfigManager::serverDescription()); + server->reloadHTTPAdvertiserConfig(); sendServerMessage("Reloaded configurations"); } diff --git a/core/src/config_manager.cpp b/core/src/config_manager.cpp index 76244c7..bbe2759 100644 --- a/core/src/config_manager.cpp +++ b/core/src/config_manager.cpp @@ -368,6 +368,21 @@ QStringList ConfigManager::gimpList() return m_commands->gimps; } +bool ConfigManager::advertiseHTTPServer() +{ + return m_settings->value("ModernAdvertiser/advertise","true").toBool(); +} + +bool ConfigManager::advertiserHTTPDebug() +{ + return m_settings->value("ModernAdvertiser/debug","true").toBool(); +} + +QUrl ConfigManager::advertiserHTTPIP() +{ + return m_settings->value("ModernAdvertiser/ms_ip","").toUrl(); +} + void ConfigManager::setMotd(const QString f_motd) { m_settings->setValue("Options/motd", f_motd); diff --git a/core/src/http_advertiser.cpp b/core/src/http_advertiser.cpp new file mode 100644 index 0000000..6763280 --- /dev/null +++ b/core/src/http_advertiser.cpp @@ -0,0 +1,82 @@ +#include "include/http_advertiser.h" + +HTTPAdvertiser::HTTPAdvertiser() +{ + m_manager = new QNetworkAccessManager(); + connect(m_manager, &QNetworkAccessManager::finished, + this, &HTTPAdvertiser::msRequestFinished); +} + +HTTPAdvertiser::~HTTPAdvertiser() +{ + m_manager->deleteLater(); +} + +void HTTPAdvertiser::msAdvertiseServer() +{ + if (m_masterserver.isValid()) { + + QUrl url(m_masterserver); + QNetworkRequest request(url); + request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json"); + + QJsonObject json; + json["port"] = m_port; + if (m_ws_port != -1) { + json["ws_port"] = m_ws_port; + } + + json["players"] = m_players; + json["name"] = m_name; + + if (!m_description.isEmpty()) { + json["description"] = m_description; + } + + m_manager->post(request, QJsonDocument(json).toJson()); + + if (m_debug) + qDebug().noquote() << "Advertised Server"; + return; + } + if (m_debug) + qWarning().noquote() << "Unable to advertise. Masterserver URL '" + m_masterserver.toString() + "' is not valid."; + return; + +} + +void HTTPAdvertiser::msRequestFinished(QNetworkReply *reply) +{ + if (m_debug) { + if (reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt() == 200) { + qDebug().noquote() << "Succesfully advertised server."; + } + else { + QJsonDocument json = QJsonDocument::fromJson(reply->readAll()); + if (json.isNull()) { + qCritical().noquote() << "Invalid JSON response from" << reply->url(); + reply->deleteLater(); + return; + } + + qDebug().noquote() << "Got valid response from" << reply->url(); + qDebug() << json; + } + } + reply->deleteLater(); +} + +void HTTPAdvertiser::setAdvertiserSettings(advertiser_config config) +{ + m_name = config.name; + m_description = config.description; + m_port = config.port; + m_ws_port = config.ws_port; + m_players = config.players; + m_masterserver = config.masterserver; + m_debug = config.debug; + + msAdvertiseServer(); +} + + diff --git a/core/src/server.cpp b/core/src/server.cpp index 635e0a9..184e031 100644 --- a/core/src/server.cpp +++ b/core/src/server.cpp @@ -29,6 +29,7 @@ Server::Server(int p_port, int p_ws_port, QObject* parent) : timer = new QTimer(); db_manager = new DBManager(); + } void Server::start() @@ -57,6 +58,18 @@ void Server::start() discord, &Discord::onModcallWebhookRequested); } + if (ConfigManager::advertiseHTTPServer()) { + httpAdvertiserTimer = new QTimer(this); + httpAdvertiser = new HTTPAdvertiser(); + + connect(httpAdvertiserTimer, &QTimer::timeout, + httpAdvertiser, &HTTPAdvertiser::msAdvertiseServer); + connect(this, &Server::reloadHTTPRequest, + httpAdvertiser, &HTTPAdvertiser::setAdvertiserSettings); + reloadHTTPAdvertiserConfig(); + httpAdvertiserTimer->start(300000); + } + proxy = new WSProxy(port, ws_port, this); if(ws_port != -1) proxy->start(); @@ -242,6 +255,19 @@ int Server::getCharID(QString char_name) return -1; // character does not exist } +void Server::reloadHTTPAdvertiserConfig() +{ + 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 reloadHTTPRequest(config); +} + void Server::allowMessage() { can_send_ic_messages = true;