fix(RD): pass copy of IndexedPack to callbacks instead of ref.

This prevents a crash in which the pack list gets updated in a search
request meanwhile a versions / extra info request is being processed.
Previously, this situation would cause the reference in the latter
callbacks to be invalidated by an internal relocation of the pack list.

Signed-off-by: flow <flowlnlnln@gmail.com>
This commit is contained in:
flow 2022-12-20 16:27:15 -03:00
parent 36571c5e22
commit 38e20eb148
No known key found for this signature in database
GPG Key ID: 8D0F221F0A59F469
2 changed files with 36 additions and 30 deletions

View File

@ -81,7 +81,7 @@ class ResourceAPI {
}; };
struct VersionSearchArgs { struct VersionSearchArgs {
ModPlatform::IndexedPack& pack; ModPlatform::IndexedPack pack;
std::optional<std::list<Version> > mcVersions; std::optional<std::list<Version> > mcVersions;
std::optional<ModLoaderTypes> loaders; std::optional<ModLoaderTypes> loaders;
@ -94,16 +94,16 @@ class ResourceAPI {
} }
}; };
struct VersionSearchCallbacks { struct VersionSearchCallbacks {
std::function<void(QJsonDocument&, ModPlatform::IndexedPack&)> on_succeed; std::function<void(QJsonDocument&, ModPlatform::IndexedPack)> on_succeed;
}; };
struct ProjectInfoArgs { struct ProjectInfoArgs {
ModPlatform::IndexedPack& pack; ModPlatform::IndexedPack pack;
void operator=(ProjectInfoArgs other) { pack = other.pack; } void operator=(ProjectInfoArgs other) { pack = other.pack; }
}; };
struct ProjectInfoCallbacks { struct ProjectInfoCallbacks {
std::function<void(QJsonDocument&, ModPlatform::IndexedPack&)> on_succeed; std::function<void(QJsonDocument&, ModPlatform::IndexedPack)> on_succeed;
}; };
public: public:

View File

@ -63,7 +63,7 @@ ResourceAPI::VersionSearchArgs ModModel::createVersionsArguments(QModelIndex& en
} }
ResourceAPI::VersionSearchCallbacks ModModel::createVersionsCallbacks(QModelIndex& entry) ResourceAPI::VersionSearchCallbacks ModModel::createVersionsCallbacks(QModelIndex& entry)
{ {
return { [this, entry](auto& doc, auto& pack) { return { [this, entry](auto& doc, auto pack) {
if (!s_running_models.constFind(this).value()) if (!s_running_models.constFind(this).value())
return; return;
versionRequestSucceeded(doc, pack, entry); versionRequestSucceeded(doc, pack, entry);
@ -77,7 +77,7 @@ ResourceAPI::ProjectInfoArgs ModModel::createInfoArguments(QModelIndex& entry)
} }
ResourceAPI::ProjectInfoCallbacks ModModel::createInfoCallbacks(QModelIndex& entry) ResourceAPI::ProjectInfoCallbacks ModModel::createInfoCallbacks(QModelIndex& entry)
{ {
return { [this, entry](auto& doc, auto& pack) { return { [this, entry](auto& doc, auto pack) {
if (!s_running_models.constFind(this).value()) if (!s_running_models.constFind(this).value())
return; return;
infoRequestFinished(doc, pack, entry); infoRequestFinished(doc, pack, entry);
@ -136,51 +136,57 @@ void ModModel::infoRequestFinished(QJsonDocument& doc, ModPlatform::IndexedPack&
{ {
qDebug() << "Loading mod info"; qDebug() << "Loading mod info";
auto current_pack = data(index, Qt::UserRole).value<ModPlatform::IndexedPack>();
// Check if the index is still valid for this mod or not
if (pack.addonId != current_pack.addonId)
return;
try { try {
auto obj = Json::requireObject(doc); auto obj = Json::requireObject(doc);
loadExtraPackInfo(pack, obj); loadExtraPackInfo(current_pack, obj);
} catch (const JSONValidationError& e) { } catch (const JSONValidationError& e) {
qDebug() << doc; qDebug() << doc;
qWarning() << "Error while reading " << debugName() << " mod info: " << e.cause(); qWarning() << "Error while reading " << debugName() << " mod info: " << e.cause();
} }
// Check if the index is still valid for this mod or not // Cache info :^)
if (pack.addonId == data(index, Qt::UserRole).value<ModPlatform::IndexedPack>().addonId) { QVariant new_pack;
// Cache info :^) new_pack.setValue(current_pack);
QVariant new_pack; if (!setData(index, new_pack, Qt::UserRole)) {
new_pack.setValue(pack); qWarning() << "Failed to cache mod info!";
if (!setData(index, new_pack, Qt::UserRole)) { return;
qWarning() << "Failed to cache mod info!";
return;
}
emit projectInfoUpdated();
} }
emit projectInfoUpdated();
} }
void ModModel::versionRequestSucceeded(QJsonDocument doc, ModPlatform::IndexedPack& pack, const QModelIndex& index) void ModModel::versionRequestSucceeded(QJsonDocument doc, ModPlatform::IndexedPack& pack, const QModelIndex& index)
{ {
auto arr = doc.isObject() ? Json::ensureArray(doc.object(), "data") : doc.array(); auto arr = doc.isObject() ? Json::ensureArray(doc.object(), "data") : doc.array();
auto current_pack = data(index, Qt::UserRole).value<ModPlatform::IndexedPack>();
// Check if the index is still valid for this mod or not
if (pack.addonId != current_pack.addonId)
return;
try { try {
loadIndexedPackVersions(pack, arr); loadIndexedPackVersions(current_pack, arr);
} catch (const JSONValidationError& e) { } catch (const JSONValidationError& e) {
qDebug() << doc; qDebug() << doc;
qWarning() << "Error while reading " << debugName() << " mod version: " << e.cause(); qWarning() << "Error while reading " << debugName() << " mod version: " << e.cause();
} }
// Check if the index is still valid for this mod or not // Cache info :^)
if (pack.addonId == data(index, Qt::UserRole).value<ModPlatform::IndexedPack>().addonId) { QVariant new_pack;
// Cache info :^) new_pack.setValue(current_pack);
QVariant new_pack; if (!setData(index, new_pack, Qt::UserRole)) {
new_pack.setValue(pack); qWarning() << "Failed to cache mod versions!";
if (!setData(index, new_pack, Qt::UserRole)) { return;
qWarning() << "Failed to cache mod versions!";
return;
}
emit versionListUpdated();
} }
emit versionListUpdated();
} }
} // namespace ResourceDownload } // namespace ResourceDownload