/* SPDX-FileCopyrightText: 2026 Project Tick * SPDX-FileContributor: Project Tick * SPDX-License-Identifier: GPL-3.0-or-later * * MeshMC - A Custom Launcher for Minecraft * Copyright (C) 2026 Project Tick * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "NotificationChecker.h" #include #include #include #include #include "net/Download.h" #include "Application.h" NotificationChecker::NotificationChecker(QObject* parent) : QObject(parent) {} void NotificationChecker::setNotificationsUrl(const QUrl& notificationsUrl) { m_notificationsUrl = notificationsUrl; } void NotificationChecker::setApplicationChannel(QString channel) { m_appVersionChannel = channel; } void NotificationChecker::setApplicationFullVersion(QString version) { m_appFullVersion = version; } void NotificationChecker::setApplicationPlatform(QString platform) { m_appPlatform = platform; } QList NotificationChecker::notificationEntries() const { return m_entries; } void NotificationChecker::checkForNotifications() { if (!m_notificationsUrl.isValid()) { qCritical() << "Failed to check for notifications. No notifications URL set." << "If you'd like to use MeshMC's notification system, please pass " "the " "URL to CMake at compile time."; return; } if (m_checkJob) { return; } m_checkJob = new NetJob("Checking for notifications", APPLICATION->network()); auto entry = APPLICATION->metacache()->resolveEntry("root", "notifications.json"); entry->setStale(true); m_checkJob->addNetAction( m_download = Net::Download::makeCached(m_notificationsUrl, entry)); connect(m_download.get(), &Net::Download::succeeded, this, &NotificationChecker::downloadSucceeded); m_checkJob->start(); } void NotificationChecker::downloadSucceeded(int) { m_entries.clear(); QFile file(m_download->getTargetFilepath()); if (file.open(QFile::ReadOnly)) { QJsonArray root = QJsonDocument::fromJson(file.readAll()).array(); for (auto it = root.begin(); it != root.end(); ++it) { QJsonObject obj = (*it).toObject(); NotificationEntry entry; entry.id = obj.value("id").toDouble(); entry.message = obj.value("message").toString(); entry.channel = obj.value("channel").toString(); entry.platform = obj.value("platform").toString(); entry.from = obj.value("from").toString(); entry.to = obj.value("to").toString(); const QString type = obj.value("type").toString("critical"); if (type == "critical") { entry.type = NotificationEntry::Critical; } else if (type == "warning") { entry.type = NotificationEntry::Warning; } else if (type == "information") { entry.type = NotificationEntry::Information; } if (entryApplies(entry)) m_entries.append(entry); } } m_checkJob.reset(); emit notificationCheckFinished(); } bool versionLessThan(const QString& v1, const QString& v2) { QStringList l1 = v1.split('.'); QStringList l2 = v2.split('.'); while (!l1.isEmpty() && !l2.isEmpty()) { int one = l1.isEmpty() ? 0 : l1.takeFirst().toInt(); int two = l2.isEmpty() ? 0 : l2.takeFirst().toInt(); if (one != two) { return one < two; } } return false; } bool NotificationChecker::entryApplies( const NotificationChecker::NotificationEntry& entry) const { bool channelApplies = entry.channel.isEmpty() || entry.channel == m_appVersionChannel; bool platformApplies = entry.platform.isEmpty() || entry.platform == m_appPlatform; bool fromApplies = entry.from.isEmpty() || entry.from == m_appFullVersion || !versionLessThan(m_appFullVersion, entry.from); bool toApplies = entry.to.isEmpty() || entry.to == m_appFullVersion || !versionLessThan(entry.to, m_appFullVersion); return channelApplies && platformApplies && fromApplies && toApplies; }