2022-05-10 19:57:47 -03:00
|
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
/*
|
2023-08-04 19:41:47 +02:00
|
|
|
* Prism Launcher - Minecraft Launcher
|
2022-09-24 00:06:36 +03:00
|
|
|
* Copyright (c) 2022 flowln <flowlnlnln@gmail.com>
|
|
|
|
*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
2022-05-10 19:57:47 -03:00
|
|
|
|
2022-04-13 19:16:36 -03:00
|
|
|
#include "Packwiz.h"
|
|
|
|
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QDir>
|
|
|
|
#include <QObject>
|
2023-08-20 00:48:46 +03:00
|
|
|
#include <sstream>
|
|
|
|
#include <string>
|
2022-04-13 19:16:36 -03:00
|
|
|
|
2022-11-03 16:59:50 -03:00
|
|
|
#include "FileSystem.h"
|
|
|
|
#include "StringUtils.h"
|
|
|
|
|
2022-04-15 22:37:10 -03:00
|
|
|
#include "minecraft/mod/Mod.h"
|
2022-04-19 20:19:51 -03:00
|
|
|
#include "modplatform/ModIndex.h"
|
2022-04-15 22:37:10 -03:00
|
|
|
|
2022-11-03 16:59:50 -03:00
|
|
|
#include <toml++/toml.h>
|
|
|
|
|
2022-04-16 13:27:29 -03:00
|
|
|
namespace Packwiz {
|
|
|
|
|
2022-05-18 05:46:07 -03:00
|
|
|
auto getRealIndexName(QDir& index_dir, QString normalized_fname, bool should_find_match) -> QString
|
|
|
|
{
|
|
|
|
QFile index_file(index_dir.absoluteFilePath(normalized_fname));
|
|
|
|
|
|
|
|
QString real_fname = normalized_fname;
|
|
|
|
if (!index_file.exists()) {
|
|
|
|
// Tries to get similar entries
|
|
|
|
for (auto& file_name : index_dir.entryList(QDir::Filter::Files)) {
|
|
|
|
if (!QString::compare(normalized_fname, file_name, Qt::CaseInsensitive)) {
|
|
|
|
real_fname = file_name;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-24 00:06:36 +03:00
|
|
|
if (should_find_match && !QString::compare(normalized_fname, real_fname, Qt::CaseSensitive)) {
|
2022-05-18 05:46:07 -03:00
|
|
|
qCritical() << "Could not find a match for a valid metadata file!";
|
|
|
|
qCritical() << "File: " << normalized_fname;
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return real_fname;
|
|
|
|
}
|
|
|
|
|
2022-04-14 22:02:41 -03:00
|
|
|
// Helpers
|
2022-06-19 14:26:15 -03:00
|
|
|
static inline auto indexFileName(QString const& mod_slug) -> QString
|
2022-04-14 22:02:41 -03:00
|
|
|
{
|
2022-09-24 00:06:36 +03:00
|
|
|
if (mod_slug.endsWith(".pw.toml"))
|
2022-06-19 14:26:15 -03:00
|
|
|
return mod_slug;
|
|
|
|
return QString("%1.pw.toml").arg(mod_slug);
|
2022-04-14 22:02:41 -03:00
|
|
|
}
|
|
|
|
|
2022-04-19 21:10:12 -03:00
|
|
|
static ModPlatform::ProviderCapabilities ProviderCaps;
|
|
|
|
|
2022-05-18 05:46:07 -03:00
|
|
|
// Helper functions for extracting data from the TOML file
|
2022-11-03 16:59:50 -03:00
|
|
|
auto stringEntry(toml::table table, QString entry_name) -> QString
|
2022-05-18 05:46:07 -03:00
|
|
|
{
|
2022-11-03 16:59:50 -03:00
|
|
|
auto node = table[StringUtils::toStdString(entry_name)];
|
2022-09-24 00:06:36 +03:00
|
|
|
if (!node) {
|
2022-11-03 16:59:50 -03:00
|
|
|
qCritical() << "Failed to read str property '" + entry_name + "' in mod metadata.";
|
2022-05-18 05:46:07 -03:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2022-11-03 16:59:50 -03:00
|
|
|
return node.value_or("");
|
2022-05-18 05:46:07 -03:00
|
|
|
}
|
|
|
|
|
2022-11-03 16:59:50 -03:00
|
|
|
auto intEntry(toml::table table, QString entry_name) -> int
|
2022-05-18 05:46:07 -03:00
|
|
|
{
|
2022-11-03 16:59:50 -03:00
|
|
|
auto node = table[StringUtils::toStdString(entry_name)];
|
2022-09-24 00:06:36 +03:00
|
|
|
if (!node) {
|
2022-11-03 16:59:50 -03:00
|
|
|
qCritical() << "Failed to read int property '" + entry_name + "' in mod metadata.";
|
2022-05-18 05:46:07 -03:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2022-09-24 00:06:36 +03:00
|
|
|
return node.value_or(0);
|
2022-05-18 05:46:07 -03:00
|
|
|
}
|
|
|
|
|
2023-06-30 23:51:15 -07:00
|
|
|
auto V1::createModFormat([[maybe_unused]] QDir& index_dir, ModPlatform::IndexedPack& mod_pack, ModPlatform::IndexedVersion& mod_version)
|
|
|
|
-> Mod
|
2022-04-13 19:16:36 -03:00
|
|
|
{
|
|
|
|
Mod mod;
|
|
|
|
|
2022-06-19 14:26:15 -03:00
|
|
|
mod.slug = mod_pack.slug;
|
2022-04-13 19:16:36 -03:00
|
|
|
mod.name = mod_pack.name;
|
|
|
|
mod.filename = mod_version.fileName;
|
|
|
|
|
2022-11-25 09:23:46 -03:00
|
|
|
if (mod_pack.provider == ModPlatform::ResourceProvider::FLAME) {
|
2022-05-07 19:39:00 -03:00
|
|
|
mod.mode = "metadata:curseforge";
|
2022-09-24 00:06:36 +03:00
|
|
|
} else {
|
2022-05-07 19:39:00 -03:00
|
|
|
mod.mode = "url";
|
|
|
|
mod.url = mod_version.downloadUrl;
|
|
|
|
}
|
|
|
|
|
2022-05-06 12:42:01 -03:00
|
|
|
mod.hash_format = mod_version.hash_type;
|
2022-04-21 15:45:20 -03:00
|
|
|
mod.hash = mod_version.hash;
|
2022-04-13 19:16:36 -03:00
|
|
|
|
|
|
|
mod.provider = mod_pack.provider;
|
2022-05-13 11:42:08 -03:00
|
|
|
mod.file_id = mod_version.fileId;
|
|
|
|
mod.project_id = mod_pack.addonId;
|
2023-10-15 11:52:28 +03:00
|
|
|
mod.side = stringToSide(mod_version.side.isEmpty() ? mod_pack.side : mod_version.side);
|
|
|
|
mod.loaders = mod_version.loaders;
|
2022-04-13 19:16:36 -03:00
|
|
|
|
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
|
2023-06-30 23:51:15 -07:00
|
|
|
auto V1::createModFormat(QDir& index_dir, [[maybe_unused]] ::Mod& internal_mod, QString slug) -> Mod
|
2022-04-15 22:37:10 -03:00
|
|
|
{
|
|
|
|
// Try getting metadata if it exists
|
2022-09-24 00:06:36 +03:00
|
|
|
Mod mod{ getIndexForMod(index_dir, slug) };
|
|
|
|
if (mod.isValid())
|
2022-04-15 22:37:10 -03:00
|
|
|
return mod;
|
|
|
|
|
2022-04-20 18:45:39 -03:00
|
|
|
qWarning() << QString("Tried to create mod metadata with a Mod without metadata!");
|
2022-04-15 22:37:10 -03:00
|
|
|
|
2022-04-20 18:45:39 -03:00
|
|
|
return {};
|
2022-04-15 22:37:10 -03:00
|
|
|
}
|
|
|
|
|
2022-04-16 13:27:29 -03:00
|
|
|
void V1::updateModIndex(QDir& index_dir, Mod& mod)
|
2022-04-13 19:16:36 -03:00
|
|
|
{
|
2022-09-24 00:06:36 +03:00
|
|
|
if (!mod.isValid()) {
|
2022-04-15 22:37:10 -03:00
|
|
|
qCritical() << QString("Tried to update metadata of an invalid mod!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-04-13 19:16:36 -03:00
|
|
|
// Ensure the corresponding mod's info exists, and create it if not
|
2022-05-18 05:46:07 -03:00
|
|
|
|
2022-06-19 14:26:15 -03:00
|
|
|
auto normalized_fname = indexFileName(mod.slug);
|
2022-05-18 05:46:07 -03:00
|
|
|
auto real_fname = getRealIndexName(index_dir, normalized_fname);
|
|
|
|
|
|
|
|
QFile index_file(index_dir.absoluteFilePath(real_fname));
|
2022-04-13 19:16:36 -03:00
|
|
|
|
2022-06-11 17:19:34 -03:00
|
|
|
if (real_fname != normalized_fname)
|
|
|
|
index_file.rename(normalized_fname);
|
|
|
|
|
2022-04-13 19:16:36 -03:00
|
|
|
// There's already data on there!
|
2022-04-15 22:37:10 -03:00
|
|
|
// TODO: We should do more stuff here, as the user is likely trying to
|
|
|
|
// override a file. In this case, check versions and ask the user what
|
|
|
|
// they want to do!
|
2022-09-24 00:06:36 +03:00
|
|
|
if (index_file.exists()) {
|
|
|
|
index_file.remove();
|
2022-11-03 16:59:50 -03:00
|
|
|
} else {
|
|
|
|
FS::ensureFilePathExists(index_file.fileName());
|
2022-09-24 00:06:36 +03:00
|
|
|
}
|
2022-04-13 19:16:36 -03:00
|
|
|
|
2023-08-26 23:36:46 +03:00
|
|
|
toml::table update;
|
|
|
|
switch (mod.provider) {
|
|
|
|
case (ModPlatform::ResourceProvider::FLAME):
|
|
|
|
if (mod.file_id.toInt() == 0 || mod.project_id.toInt() == 0) {
|
|
|
|
qCritical() << QString("Did not write file %1 because missing information!").arg(normalized_fname);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
update = toml::table{
|
|
|
|
{ "file-id", mod.file_id.toInt() },
|
|
|
|
{ "project-id", mod.project_id.toInt() },
|
|
|
|
};
|
|
|
|
break;
|
|
|
|
case (ModPlatform::ResourceProvider::MODRINTH):
|
|
|
|
if (mod.mod_id().toString().isEmpty() || mod.version().toString().isEmpty()) {
|
|
|
|
qCritical() << QString("Did not write file %1 because missing information!").arg(normalized_fname);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
update = toml::table{
|
|
|
|
{ "mod-id", mod.mod_id().toString().toStdString() },
|
|
|
|
{ "version", mod.version().toString().toStdString() },
|
|
|
|
};
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2023-10-15 11:52:28 +03:00
|
|
|
toml::array loaders;
|
|
|
|
for (auto loader : { ModPlatform::NeoForge, ModPlatform::Forge, ModPlatform::Cauldron, ModPlatform::LiteLoader, ModPlatform::Fabric,
|
|
|
|
ModPlatform::Quilt }) {
|
|
|
|
if (mod.loaders & loader) {
|
|
|
|
loaders.push_back(getModLoaderAsString(loader));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-13 19:16:36 -03:00
|
|
|
if (!index_file.open(QIODevice::ReadWrite)) {
|
2023-08-26 23:36:46 +03:00
|
|
|
qCritical() << QString("Could not open file %1!").arg(normalized_fname);
|
2022-04-13 19:16:36 -03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Put TOML data into the file
|
|
|
|
QTextStream in_stream(&index_file);
|
|
|
|
{
|
2023-08-20 00:48:46 +03:00
|
|
|
auto tbl = toml::table{ { "name", mod.name.toStdString() },
|
|
|
|
{ "filename", mod.filename.toStdString() },
|
2023-09-10 16:22:57 +03:00
|
|
|
{ "side", sideToString(mod.side).toStdString() },
|
2023-10-15 11:52:28 +03:00
|
|
|
{ "loader", loaders },
|
2023-08-20 00:48:46 +03:00
|
|
|
{ "download",
|
|
|
|
toml::table{
|
|
|
|
{ "mode", mod.mode.toStdString() },
|
|
|
|
{ "url", mod.url.toString().toStdString() },
|
|
|
|
{ "hash-format", mod.hash_format.toStdString() },
|
|
|
|
{ "hash", mod.hash.toStdString() },
|
|
|
|
} },
|
|
|
|
{ "update", toml::table{ { ProviderCaps.name(mod.provider), update } } } };
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << tbl;
|
|
|
|
in_stream << QString::fromStdString(ss.str());
|
2022-04-13 19:16:36 -03:00
|
|
|
}
|
2022-05-18 05:46:07 -03:00
|
|
|
|
2022-06-19 14:26:15 -03:00
|
|
|
index_file.flush();
|
2022-05-18 05:46:07 -03:00
|
|
|
index_file.close();
|
2022-04-13 19:16:36 -03:00
|
|
|
}
|
2022-04-13 21:25:08 -03:00
|
|
|
|
2022-06-19 14:26:15 -03:00
|
|
|
void V1::deleteModIndex(QDir& index_dir, QString& mod_slug)
|
2022-04-14 22:02:41 -03:00
|
|
|
{
|
2022-06-19 14:26:15 -03:00
|
|
|
auto normalized_fname = indexFileName(mod_slug);
|
2022-05-18 05:46:07 -03:00
|
|
|
auto real_fname = getRealIndexName(index_dir, normalized_fname);
|
|
|
|
if (real_fname.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
QFile index_file(index_dir.absoluteFilePath(real_fname));
|
2022-04-14 22:02:41 -03:00
|
|
|
|
2022-06-11 17:19:34 -03:00
|
|
|
if (!index_file.exists()) {
|
2022-06-19 14:26:15 -03:00
|
|
|
qWarning() << QString("Tried to delete non-existent mod metadata for %1!").arg(mod_slug);
|
2022-04-14 22:02:41 -03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-06-11 17:19:34 -03:00
|
|
|
if (!index_file.remove()) {
|
2022-06-19 14:26:15 -03:00
|
|
|
qWarning() << QString("Failed to remove metadata for mod %1!").arg(mod_slug);
|
2022-04-14 22:02:41 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-11 17:19:34 -03:00
|
|
|
void V1::deleteModIndex(QDir& index_dir, QVariant& mod_id)
|
|
|
|
{
|
|
|
|
for (auto& file_name : index_dir.entryList(QDir::Filter::Files)) {
|
|
|
|
auto mod = getIndexForMod(index_dir, file_name);
|
|
|
|
|
|
|
|
if (mod.mod_id() == mod_id) {
|
|
|
|
deleteModIndex(index_dir, mod.name);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-19 14:26:15 -03:00
|
|
|
auto V1::getIndexForMod(QDir& index_dir, QString slug) -> Mod
|
2022-04-13 21:25:08 -03:00
|
|
|
{
|
|
|
|
Mod mod;
|
|
|
|
|
2022-06-19 14:26:15 -03:00
|
|
|
auto normalized_fname = indexFileName(slug);
|
2022-05-18 05:46:07 -03:00
|
|
|
auto real_fname = getRealIndexName(index_dir, normalized_fname, true);
|
|
|
|
if (real_fname.isEmpty())
|
2022-04-15 22:37:10 -03:00
|
|
|
return {};
|
2022-05-18 05:46:07 -03:00
|
|
|
|
2022-09-24 00:06:36 +03:00
|
|
|
toml::table table;
|
|
|
|
#if TOML_EXCEPTIONS
|
|
|
|
try {
|
2022-11-03 16:59:50 -03:00
|
|
|
table = toml::parse_file(StringUtils::toStdString(index_dir.absoluteFilePath(real_fname)));
|
2022-09-24 00:06:36 +03:00
|
|
|
} catch (const toml::parse_error& err) {
|
|
|
|
qWarning() << QString("Could not open file %1!").arg(normalized_fname);
|
|
|
|
qWarning() << "Reason: " << QString(err.what());
|
2022-04-15 22:37:10 -03:00
|
|
|
return {};
|
2022-04-14 22:02:41 -03:00
|
|
|
}
|
2022-09-24 00:06:36 +03:00
|
|
|
#else
|
2023-07-01 13:38:32 -07:00
|
|
|
toml::parse_result result = toml::parse_file(StringUtils::toStdString(index_dir.absoluteFilePath(real_fname)));
|
|
|
|
if (!result) {
|
2022-06-19 14:26:15 -03:00
|
|
|
qWarning() << QString("Could not open file %1!").arg(normalized_fname);
|
2023-07-01 13:38:32 -07:00
|
|
|
qWarning() << "Reason: " << result.error().description();
|
2022-04-15 22:37:10 -03:00
|
|
|
return {};
|
2022-04-13 21:25:08 -03:00
|
|
|
}
|
2023-07-01 13:38:32 -07:00
|
|
|
table = result.table();
|
2022-09-24 00:06:36 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// index_file.close();
|
2022-05-18 05:46:07 -03:00
|
|
|
|
2022-06-19 14:26:15 -03:00
|
|
|
mod.slug = slug;
|
|
|
|
|
2022-05-18 05:46:07 -03:00
|
|
|
{ // Basic info
|
2022-04-13 21:25:08 -03:00
|
|
|
mod.name = stringEntry(table, "name");
|
|
|
|
mod.filename = stringEntry(table, "filename");
|
2023-09-10 16:22:57 +03:00
|
|
|
mod.side = stringToSide(stringEntry(table, "side"));
|
2023-10-15 11:52:28 +03:00
|
|
|
if (auto loaders = table["loaders"]; loaders && loaders.is_array()) {
|
|
|
|
for (auto&& loader : *loaders.as_array()) {
|
|
|
|
if (loader.is_string()) {
|
|
|
|
mod.loaders |= ModPlatform::getModLoaderFromString(QString::fromStdString(loader.as_string()->value_or("")));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-04-13 21:25:08 -03:00
|
|
|
}
|
|
|
|
|
2022-05-18 05:46:07 -03:00
|
|
|
{ // [download] info
|
2022-09-24 00:06:36 +03:00
|
|
|
auto download_table = table["download"].as_table();
|
2022-04-13 21:25:08 -03:00
|
|
|
if (!download_table) {
|
|
|
|
qCritical() << QString("No [download] section found on mod metadata!");
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2022-09-24 00:06:36 +03:00
|
|
|
mod.mode = stringEntry(*download_table, "mode");
|
|
|
|
mod.url = stringEntry(*download_table, "url");
|
|
|
|
mod.hash_format = stringEntry(*download_table, "hash-format");
|
|
|
|
mod.hash = stringEntry(*download_table, "hash");
|
2022-04-13 21:25:08 -03:00
|
|
|
}
|
|
|
|
|
2022-09-24 00:06:36 +03:00
|
|
|
{ // [update] info
|
2022-11-25 09:23:46 -03:00
|
|
|
using Provider = ModPlatform::ResourceProvider;
|
2022-04-13 21:25:08 -03:00
|
|
|
|
2022-09-24 00:06:36 +03:00
|
|
|
auto update_table = table["update"];
|
|
|
|
if (!update_table || !update_table.is_table()) {
|
2022-04-13 21:25:08 -03:00
|
|
|
qCritical() << QString("No [update] section found on mod metadata!");
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2022-09-24 00:06:36 +03:00
|
|
|
toml::table* mod_provider_table = nullptr;
|
|
|
|
if ((mod_provider_table = update_table[ProviderCaps.name(Provider::FLAME)].as_table())) {
|
2022-04-13 21:25:08 -03:00
|
|
|
mod.provider = Provider::FLAME;
|
2022-09-24 00:06:36 +03:00
|
|
|
mod.file_id = intEntry(*mod_provider_table, "file-id");
|
|
|
|
mod.project_id = intEntry(*mod_provider_table, "project-id");
|
|
|
|
} else if ((mod_provider_table = update_table[ProviderCaps.name(Provider::MODRINTH)].as_table())) {
|
2022-04-13 21:25:08 -03:00
|
|
|
mod.provider = Provider::MODRINTH;
|
2022-09-24 00:06:36 +03:00
|
|
|
mod.mod_id() = stringEntry(*mod_provider_table, "mod-id");
|
|
|
|
mod.version() = stringEntry(*mod_provider_table, "version");
|
2022-04-13 21:25:08 -03:00
|
|
|
} else {
|
2022-04-14 22:02:41 -03:00
|
|
|
qCritical() << QString("No mod provider on mod metadata!");
|
2022-04-13 21:25:08 -03:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return mod;
|
|
|
|
}
|
2022-04-16 13:27:29 -03:00
|
|
|
|
2022-06-11 17:19:34 -03:00
|
|
|
auto V1::getIndexForMod(QDir& index_dir, QVariant& mod_id) -> Mod
|
|
|
|
{
|
|
|
|
for (auto& file_name : index_dir.entryList(QDir::Filter::Files)) {
|
|
|
|
auto mod = getIndexForMod(index_dir, file_name);
|
|
|
|
|
|
|
|
if (mod.mod_id() == mod_id)
|
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-09-10 16:22:57 +03:00
|
|
|
auto V1::sideToString(Side side) -> QString
|
|
|
|
{
|
|
|
|
switch (side) {
|
|
|
|
case Side::ClientSide:
|
|
|
|
return "client";
|
|
|
|
case Side::ServerSide:
|
|
|
|
return "server";
|
|
|
|
case Side::UniversalSide:
|
|
|
|
return "both";
|
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
auto V1::stringToSide(QString side) -> Side
|
|
|
|
{
|
|
|
|
if (side == "client")
|
|
|
|
return Side::ClientSide;
|
|
|
|
if (side == "server")
|
|
|
|
return Side::ServerSide;
|
|
|
|
if (side == "both")
|
|
|
|
return Side::UniversalSide;
|
|
|
|
return Side::UniversalSide;
|
|
|
|
}
|
|
|
|
|
2022-06-11 17:19:34 -03:00
|
|
|
} // namespace Packwiz
|