atrooney-online-2/src/networkmanager.cpp
Rosemary Witchaven e36d79f749
Fix or suppress compiler warnings (Qt 5.15 w/MinGW) (#818)
* Major cleanup of screenshake code

* Add pre-5.10 support for screenshake math

* more compat, uglier too

* add surprise tool

* we don't need inline functions

* only run qsrand on old versions

* Squash compiler warnings

* >= not > please

* only run qsrand on old versions

* Squash compiler warnings

* >= not > please

Co-authored-by: stonedDiscord <Tukz@gmx.de>
Co-authored-by: Salanto <62221668+Salanto@users.noreply.github.com>
2022-07-29 18:23:36 +02:00

255 lines
7.7 KiB
C++

#include "networkmanager.h"
#include "datatypes.h"
#include "debug_functions.h"
#include "lobby.h"
#include <QAbstractSocket>
#include <QJsonArray>
#include <QJsonDocument>
#include <QNetworkReply>
NetworkManager::NetworkManager(AOApplication *parent) : QObject(parent)
{
ao_app = parent;
http = new QNetworkAccessManager(this);
heartbeat_timer = new QTimer(this);
QString master_config =
ao_app->configini->value("master", "").value<QString>();
if (!master_config.isEmpty() && QUrl(master_config).scheme().startsWith("http")) {
qInfo() << "using alternate master server" << master_config;
ms_baseurl = master_config;
}
connect(heartbeat_timer, &QTimer::timeout, this, &NetworkManager::send_heartbeat);
heartbeat_timer->start(heartbeat_interval);
}
void NetworkManager::get_server_list(const std::function<void()> &cb)
{
QNetworkRequest req(QUrl(ms_baseurl + "/servers"));
req.setRawHeader("User-Agent", get_user_agent().toUtf8());
QNetworkReply *reply = http->get(req);
connect(reply, &QNetworkReply::finished,
this, std::bind(&NetworkManager::ms_request_finished, this, reply, cb));
}
void NetworkManager::ms_request_finished(QNetworkReply *reply,
const std::function<void()> &cb)
{
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();
QVector<server_type> server_list;
const auto jsonEntries = json.array();
for (const auto &entryRef : jsonEntries) {
const auto entry = entryRef.toObject();
server_type server;
server.ip = entry["ip"].toString();
server.name = entry["name"].toString();
server.desc = entry["description"].toString(tr("No description provided."));
if (entry["ws_port"].isDouble()) {
server.socket_type = WEBSOCKETS;
server.port = entry["ws_port"].toInt();
} else {
server.socket_type = TCP;
server.port = entry["port"].toInt();
}
if (server.port != 0) {
server_list.append(server);
}
}
ao_app->set_server_list(server_list);
cb();
reply->deleteLater();
}
void NetworkManager::send_heartbeat()
{
// Ping the server periodically to tell the MS that you've been playing
// within a 5 minute window, so that the the number of people playing within
// that time period can be counted and an accurate player count be displayed.
// What do I care about your personal information, I really don't want it.
if (ao_app->get_player_count_optout())
return;
QNetworkRequest req(QUrl(ms_baseurl + "/playing"));
req.setRawHeader("User-Agent", get_user_agent().toUtf8());
http->post(req, QByteArray());
}
void NetworkManager::request_document(MSDocumentType document_type,
const std::function<void(QString)> &cb)
{
const QMap<MSDocumentType, QString> endpoints {
// I have to balance an evil with a good
{ MSDocumentType::PrivacyPolicy, "/privacy" },
{ MSDocumentType::Motd, "/motd" },
{ MSDocumentType::ClientVersion, "/version" }
};
const QString &endpoint = endpoints[document_type];
QNetworkRequest req(QUrl(ms_baseurl + endpoint));
req.setRawHeader("User-Agent", get_user_agent().toUtf8());
QString language =
ao_app->configini->value("language").toString();
if (language.trimmed().isEmpty())
language = QLocale::system().name();
req.setRawHeader("Accept-Language", language.toUtf8());
qDebug().noquote().nospace()
<< "Getting " << endpoint << ", Accept-Language: " << language;
QNetworkReply *reply = http->get(req);
connect(reply, &QNetworkReply::finished, this, [endpoint, cb, reply] {
QString content = QString::fromUtf8(reply->readAll());
int http_status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
if (content.isEmpty() || http_status != 200) {
qDebug().noquote().nospace()
<< "Failed to get " << endpoint << " (" << reply->errorString() << ") "
<< "(http status " << http_status << ")";
content = QString();
}
cb(content);
reply->deleteLater();
});
}
void NetworkManager::connect_to_server(server_type p_server)
{
disconnect_from_server();
qInfo().nospace().noquote() << "connecting to " << p_server.ip << ":"
<< p_server.port;
switch (p_server.socket_type) {
default:
p_server.socket_type = TCP;
[[fallthrough]];
case TCP:
qInfo() << "using TCP backend";
server_socket.tcp = new QTcpSocket(this);
connect(server_socket.tcp, &QAbstractSocket::connected, this, [] {
qDebug() << "established connection to server";
});
connect(server_socket.tcp, &QIODevice::readyRead, this, [this] {
handle_server_packet(QString::fromUtf8(server_socket.tcp->readAll()));
});
connect(server_socket.tcp, &QAbstractSocket::disconnected, ao_app,
&AOApplication::server_disconnected);
#if QT_VERSION < QT_VERSION_CHECK(5, 15, 0)
connect(server_socket.tcp, QOverload<QAbstractSocket::SocketError>::of(&QAbstractSocket::error), this, [this] {
#else
connect(server_socket.tcp, &QAbstractSocket::errorOccurred, this, [this] {
#endif
qCritical() << "TCP socket error:" << server_socket.tcp->errorString();
});
server_socket.tcp->connectToHost(p_server.ip, p_server.port);
break;
case WEBSOCKETS:
qInfo() << "using WebSockets backend";
server_socket.ws = new QWebSocket(QString(), QWebSocketProtocol::VersionLatest, this);
connect(server_socket.ws, &QWebSocket::connected, this, [] {
qDebug() << "established connection to server";
});
connect(server_socket.ws, &QWebSocket::textMessageReceived, this,
&NetworkManager::handle_server_packet);
connect(server_socket.ws, &QWebSocket::disconnected, ao_app,
&AOApplication::server_disconnected);
connect(server_socket.ws, QOverload<QAbstractSocket::SocketError>::of(&QWebSocket::error),
this, [this] {
qCritical() << "WebSockets error:" << server_socket.ws->errorString();
});
QUrl url;
url.setScheme("ws");
url.setHost(p_server.ip);
url.setPort(p_server.port);
QNetworkRequest req(url);
req.setHeader(QNetworkRequest::UserAgentHeader, get_user_agent());
server_socket.ws->open(req);
break;
}
connected = true;
active_connection_type = p_server.socket_type;
}
void NetworkManager::disconnect_from_server()
{
if (!connected)
return;
switch (active_connection_type) {
case TCP:
server_socket.tcp->close();
server_socket.tcp->deleteLater();
break;
case WEBSOCKETS:
server_socket.ws->close(QWebSocketProtocol::CloseCodeGoingAway);
server_socket.ws->deleteLater();
break;
}
connected = false;
}
void NetworkManager::ship_server_packet(QString p_packet)
{
switch (active_connection_type) {
case TCP:
server_socket.tcp->write(p_packet.toUtf8());
break;
case WEBSOCKETS:
server_socket.ws->sendTextMessage(p_packet);
break;
}
}
void NetworkManager::handle_server_packet(const QString& p_data)
{
QString in_data = p_data;
if (!p_data.endsWith("%")) {
partial_packet = true;
temp_packet += in_data;
return;
}
else {
if (partial_packet) {
in_data = temp_packet + in_data;
temp_packet = "";
partial_packet = false;
}
}
#if QT_VERSION < QT_VERSION_CHECK(5, 14, 0)
const QStringList packet_list = in_data.split("%", QString::SkipEmptyParts);
#else
const QStringList packet_list = in_data.split("%", Qt::SkipEmptyParts);
#endif
for (const QString &packet : packet_list) {
AOPacket *f_packet = new AOPacket(packet);
ao_app->server_packet_received(f_packet);
}
}