2022-05-26 23:18:54 +02:00
|
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
/*
|
2023-08-04 19:41:47 +02:00
|
|
|
* Prism Launcher - Minecraft Launcher
|
2022-05-26 23:18:54 +02:00
|
|
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
|
|
|
*
|
|
|
|
* 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, version 3.
|
|
|
|
*
|
|
|
|
* 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 <https://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* This file incorporates work covered by the following copyright and
|
|
|
|
* permission notice:
|
|
|
|
*
|
|
|
|
* Copyright 2013-2021 MultiMC Contributors
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2015-01-27 22:31:07 +01:00
|
|
|
#include "ProfileUtils.h"
|
2015-02-02 02:14:14 +01:00
|
|
|
#include <QDebug>
|
2023-08-14 18:16:53 +02:00
|
|
|
#include "Json.h"
|
|
|
|
#include "minecraft/OneSixVersionFormat.h"
|
|
|
|
#include "minecraft/VersionFilterData.h"
|
2015-01-27 22:31:07 +01:00
|
|
|
|
|
|
|
#include <QJsonArray>
|
2023-08-14 18:16:53 +02:00
|
|
|
#include <QJsonDocument>
|
2015-01-27 22:31:07 +01:00
|
|
|
#include <QRegularExpression>
|
2015-02-28 10:20:25 +01:00
|
|
|
#include <QSaveFile>
|
2015-01-27 22:31:07 +01:00
|
|
|
|
2023-08-14 18:16:53 +02:00
|
|
|
namespace ProfileUtils {
|
2015-01-27 22:31:07 +01:00
|
|
|
|
|
|
|
static const int currentOrderFileVersion = 1;
|
|
|
|
|
2023-08-14 18:16:53 +02:00
|
|
|
bool readOverrideOrders(QString path, PatchOrder& order)
|
2015-01-27 22:31:07 +01:00
|
|
|
{
|
|
|
|
QFile orderFile(path);
|
2023-08-14 18:16:53 +02:00
|
|
|
if (!orderFile.exists()) {
|
2015-02-02 02:14:14 +01:00
|
|
|
qWarning() << "Order file doesn't exist. Ignoring.";
|
2015-01-27 22:31:07 +01:00
|
|
|
return false;
|
|
|
|
}
|
2023-08-14 18:16:53 +02:00
|
|
|
if (!orderFile.open(QFile::ReadOnly)) {
|
|
|
|
qCritical() << "Couldn't open" << orderFile.fileName() << " for reading:" << orderFile.errorString();
|
2015-02-02 02:14:14 +01:00
|
|
|
qWarning() << "Ignoring overriden order";
|
2015-01-27 22:31:07 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// and it's valid JSON
|
|
|
|
QJsonParseError error;
|
|
|
|
QJsonDocument doc = QJsonDocument::fromJson(orderFile.readAll(), &error);
|
2023-08-14 18:16:53 +02:00
|
|
|
if (error.error != QJsonParseError::NoError) {
|
2015-02-02 02:14:14 +01:00
|
|
|
qCritical() << "Couldn't parse" << orderFile.fileName() << ":" << error.errorString();
|
|
|
|
qWarning() << "Ignoring overriden order";
|
2015-01-27 22:31:07 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// and then read it and process it if all above is true.
|
2023-08-14 18:16:53 +02:00
|
|
|
try {
|
2015-06-03 21:57:22 +02:00
|
|
|
auto obj = Json::requireObject(doc);
|
2015-01-27 22:31:07 +01:00
|
|
|
// check order file version.
|
2015-06-03 21:57:22 +02:00
|
|
|
auto version = Json::requireInteger(obj.value("version"));
|
2023-08-14 18:16:53 +02:00
|
|
|
if (version != currentOrderFileVersion) {
|
|
|
|
throw JSONValidationError(QObject::tr("Invalid order file version, expected %1").arg(currentOrderFileVersion));
|
2015-01-27 22:31:07 +01:00
|
|
|
}
|
2015-06-03 21:57:22 +02:00
|
|
|
auto orderArray = Json::requireArray(obj.value("order"));
|
2023-08-14 18:16:53 +02:00
|
|
|
for (auto item : orderArray) {
|
2015-06-03 21:57:22 +02:00
|
|
|
order.append(Json::requireString(item));
|
2015-01-27 22:31:07 +01:00
|
|
|
}
|
2023-08-14 18:16:53 +02:00
|
|
|
} catch ([[maybe_unused]] const JSONValidationError& err) {
|
2015-02-02 02:14:14 +01:00
|
|
|
qCritical() << "Couldn't parse" << orderFile.fileName() << ": bad file format";
|
|
|
|
qWarning() << "Ignoring overriden order";
|
2015-01-27 22:31:07 +01:00
|
|
|
order.clear();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-03-18 15:02:54 +01:00
|
|
|
static VersionFilePtr createErrorVersionFile(QString fileId, QString filepath, QString error)
|
|
|
|
{
|
|
|
|
auto outError = std::make_shared<VersionFile>();
|
2017-03-19 23:58:54 +01:00
|
|
|
outError->uid = outError->name = fileId;
|
2017-03-27 03:34:39 +02:00
|
|
|
// outError->filename = filepath;
|
|
|
|
outError->addProblem(ProblemSeverity::Error, error);
|
2016-03-18 15:02:54 +01:00
|
|
|
return outError;
|
|
|
|
}
|
|
|
|
|
2023-08-14 18:16:53 +02:00
|
|
|
static VersionFilePtr guardedParseJson(const QJsonDocument& doc, const QString& fileId, const QString& filepath, const bool& requireOrder)
|
2016-03-18 15:02:54 +01:00
|
|
|
{
|
2023-08-14 18:16:53 +02:00
|
|
|
try {
|
2016-03-18 15:02:54 +01:00
|
|
|
return OneSixVersionFormat::versionFileFromJson(doc, filepath, requireOrder);
|
2023-08-14 18:16:53 +02:00
|
|
|
} catch (const Exception& e) {
|
2016-03-18 15:02:54 +01:00
|
|
|
return createErrorVersionFile(fileId, filepath, e.cause());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-14 18:16:53 +02:00
|
|
|
VersionFilePtr parseJsonFile(const QFileInfo& fileInfo, const bool requireOrder)
|
2015-01-27 22:31:07 +01:00
|
|
|
{
|
|
|
|
QFile file(fileInfo.absoluteFilePath());
|
2023-08-14 18:16:53 +02:00
|
|
|
if (!file.open(QFile::ReadOnly)) {
|
2016-03-18 15:02:54 +01:00
|
|
|
auto errorStr = QObject::tr("Unable to open the version file %1: %2.").arg(fileInfo.fileName(), file.errorString());
|
|
|
|
return createErrorVersionFile(fileInfo.completeBaseName(), fileInfo.absoluteFilePath(), errorStr);
|
2015-01-27 22:31:07 +01:00
|
|
|
}
|
|
|
|
QJsonParseError error;
|
2016-02-21 01:44:27 +01:00
|
|
|
auto data = file.readAll();
|
|
|
|
QJsonDocument doc = QJsonDocument::fromJson(data, &error);
|
2016-03-18 15:02:54 +01:00
|
|
|
file.close();
|
2023-08-14 18:16:53 +02:00
|
|
|
if (error.error != QJsonParseError::NoError) {
|
2016-03-18 15:02:54 +01:00
|
|
|
int line = 1;
|
2016-02-21 01:44:27 +01:00
|
|
|
int column = 0;
|
2023-08-14 18:16:53 +02:00
|
|
|
for (int i = 0; i < error.offset; i++) {
|
|
|
|
if (data[i] == '\n') {
|
2016-02-21 01:44:27 +01:00
|
|
|
line++;
|
|
|
|
column = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
column++;
|
|
|
|
}
|
2016-03-18 15:02:54 +01:00
|
|
|
auto errorStr = QObject::tr("Unable to process the version file %1: %2 at line %3 column %4.")
|
2023-08-14 18:16:53 +02:00
|
|
|
.arg(fileInfo.fileName(), error.errorString())
|
|
|
|
.arg(line)
|
|
|
|
.arg(column);
|
2016-03-18 15:02:54 +01:00
|
|
|
return createErrorVersionFile(fileInfo.completeBaseName(), fileInfo.absoluteFilePath(), errorStr);
|
2015-01-27 22:31:07 +01:00
|
|
|
}
|
2016-03-18 15:02:54 +01:00
|
|
|
return guardedParseJson(doc, fileInfo.completeBaseName(), fileInfo.absoluteFilePath(), requireOrder);
|
2015-01-27 22:31:07 +01:00
|
|
|
}
|
|
|
|
|
2023-08-14 18:16:53 +02:00
|
|
|
bool saveJsonFile(const QJsonDocument doc, const QString& filename)
|
2017-11-11 01:38:31 +01:00
|
|
|
{
|
|
|
|
auto data = doc.toJson();
|
|
|
|
QSaveFile jsonFile(filename);
|
2023-08-14 18:16:53 +02:00
|
|
|
if (!jsonFile.open(QIODevice::WriteOnly)) {
|
2017-11-11 01:38:31 +01:00
|
|
|
jsonFile.cancelWriting();
|
|
|
|
qWarning() << "Couldn't open" << filename << "for writing";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
jsonFile.write(data);
|
2023-08-14 18:16:53 +02:00
|
|
|
if (!jsonFile.commit()) {
|
2017-11-11 01:38:31 +01:00
|
|
|
qWarning() << "Couldn't save" << filename;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-01-27 22:31:07 +01:00
|
|
|
void removeLwjglFromPatch(VersionFilePtr patch)
|
|
|
|
{
|
2023-08-14 18:16:53 +02:00
|
|
|
auto filter = [](QList<LibraryPtr>& libs) {
|
2016-03-07 02:01:28 +01:00
|
|
|
QList<LibraryPtr> filteredLibs;
|
2023-08-14 18:16:53 +02:00
|
|
|
for (auto lib : libs) {
|
|
|
|
if (!g_VersionFilterData.lwjglWhitelist.contains(lib->artifactPrefix())) {
|
2015-01-27 22:31:07 +01:00
|
|
|
filteredLibs.append(lib);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
libs = filteredLibs;
|
|
|
|
};
|
2016-03-13 02:28:55 +01:00
|
|
|
filter(patch->libraries);
|
2015-01-27 22:31:07 +01:00
|
|
|
}
|
2023-08-14 18:16:53 +02:00
|
|
|
} // namespace ProfileUtils
|