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..05ee05c 100644
--- a/core/include/config_manager.h
+++ b/core/include/config_manager.h
@@ -302,6 +302,22 @@ class ConfigManager {
*/
static QStringList gimpList();
+ /**
+ * @brief advertise_server HTTP advertiser is not constructed if this is false.
+ */
+ static bool advertiseHTTPServer();
+
+ /**
+ * @brief advertise_debug Allows HTTP advertiser to print debug info.
+ */
+ static bool advertiserHTTPDebug();
+
+ /**
+ * @brief advertiser_ip 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..96b44df
--- /dev/null
+++ b/core/include/http_advertiser.h
@@ -0,0 +1,105 @@
+//////////////////////////////////////////////////////////////////////////////////////
+// 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
+
+
+/**
+ * @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();
+
+public slots:
+
+ /**
+ * @brief msAdvertiseServer Establishes a connection with masterserver to
+ * register or update the listing on the masterserver.
+ */
+ void msAdvertiseServer();
+
+ /**
+ * @brief msRequestFinished Reads the information send as a reply for further
+ * error handling.
+ */
+ void msRequestFinished(QNetworkReply *reply);
+
+ /**
+ * @brief setAdvertiserSettings Configures the values being advertised to masterserver.
+ * @param f_name Servername.
+ * @param f_description Serverdescription.
+ * @param f_port Client port.
+ * @param f_ws_port Optional Websocket proxy port.
+ * @param f_players Maximum amount of clients.
+ * @param f_master_url URL of the advertisement target.
+ */
+ void setAdvertiserSettings(QString f_name, QString f_description, int f_port, int f_ws_port, int f_players, QUrl f_master_url, bool f_debug);
+
+private:
+
+ /**
+ * @brief m_manager NetworkAccessManager for HTTP Advertiser.
+ */
+ QNetworkAccessManager* m_manager;
+
+ /**
+ * @brief m_name Current name of the server.
+ */
+ QString m_name;
+
+ /**
+ * @brief m_description Current description of the server.
+ */
+ QString m_description;
+
+ /**
+ * @brief m_port Current port for AO2-Clients.
+ */
+ int m_port;
+
+ /**
+ * @brief m_ws_port Websocket proxy port for WebAO-Clients.
+ */
+ int m_ws_port;
+
+ /**
+ * @brief m_players Maximum number of connected clients.
+ */
+ int m_players;
+
+ /**
+ * @brief m_masterserver URL of the masterserver being advertised to.
+ */
+ QUrl m_masterserver;
+
+ /**
+ * @brief m_debug If debug information is displayed in console.
+ */
+ bool m_debug;
+};
+
+#endif // HTTP_ADVERTISER_H
diff --git a/core/include/server.h b/core/include/server.h
index 05e1f48..72a79c8 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
@@ -227,6 +228,17 @@ class Server : public QObject {
*/
void reloadRequest(QString p_name, QString p_desc);
+ /**
+ * @brief Sends all necessary info for the new advertiser.
+ * @param f_name Servername.
+ * @param f_description Serverdescription.
+ * @param f_port Client port.
+ * @param f_ws_port Optional Websocket proxy port.
+ * @param f_players Maximum amount of clients.
+ * @param f_master_url URL of the advertisement target.
+ */
+ void reloadHTTPRequest(QString f_name, QString f_description, int f_port, int f_ws_port, int f_players, QUrl f_master_url, bool f_debug);
+
/**
* @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..3ac965f 100644
--- a/core/src/commands/moderation.cpp
+++ b/core/src/commands/moderation.cpp
@@ -412,6 +412,9 @@ void AOClient::cmdReload(int argc, QStringList argv)
{
ConfigManager::reloadSettings();
emit server->reloadRequest(ConfigManager::serverName(), ConfigManager::serverDescription());
+ emit server->reloadHTTPRequest(ConfigManager::serverName(),ConfigManager::serverDescription(),ConfigManager::serverPort(),
+ ConfigManager::webaoPort(),ConfigManager::maxPlayers(),ConfigManager::advertiserHTTPIP(),
+ ConfigManager::advertiserHTTPDebug());
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..93c12ab
--- /dev/null
+++ b/core/src/http_advertiser.cpp
@@ -0,0 +1,69 @@
+#include "include/http_advertiser.h"
+
+HTTPAdvertiser::HTTPAdvertiser()
+{
+ m_manager = new QNetworkAccessManager();
+ connect(m_manager, &QNetworkAccessManager::finished,
+ this, &HTTPAdvertiser::msRequestFinished);
+}
+
+void HTTPAdvertiser::msAdvertiseServer()
+{
+ if (m_masterserver.isValid()) {
+
+ QNetworkRequest request((QUrl (m_masterserver)));
+ 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) {
+ 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(QString f_name, QString f_description, int f_port, int f_ws_port, int f_players, QUrl f_master_url, bool f_debug)
+{
+ m_name = f_name;
+ m_description = f_description;
+ m_port = f_port;
+ m_ws_port = f_ws_port;
+ m_players = f_players;
+ m_masterserver = f_master_url;
+ m_debug = f_debug;
+
+ msAdvertiseServer();
+}
diff --git a/core/src/server.cpp b/core/src/server.cpp
index 635e0a9..cc9cbf6 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,20 @@ 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);
+
+ emit reloadHTTPRequest(ConfigManager::serverName(),ConfigManager::serverDescription(),ConfigManager::serverPort(),ConfigManager::webaoPort(),
+ ConfigManager::maxPlayers(),ConfigManager::advertiserHTTPIP(),ConfigManager::advertiserHTTPDebug());
+ httpAdvertiserTimer->start(300000);
+ }
+
proxy = new WSProxy(port, ws_port, this);
if(ws_port != -1)
proxy->start();