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:
parent
36571c5e22
commit
38e20eb148
@ -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:
|
||||||
|
@ -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
|
|
||||||
if (pack.addonId == data(index, Qt::UserRole).value<ModPlatform::IndexedPack>().addonId) {
|
|
||||||
// Cache info :^)
|
// Cache info :^)
|
||||||
QVariant new_pack;
|
QVariant new_pack;
|
||||||
new_pack.setValue(pack);
|
new_pack.setValue(current_pack);
|
||||||
if (!setData(index, new_pack, Qt::UserRole)) {
|
if (!setData(index, new_pack, Qt::UserRole)) {
|
||||||
qWarning() << "Failed to cache mod info!";
|
qWarning() << "Failed to cache mod info!";
|
||||||
return;
|
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
|
|
||||||
if (pack.addonId == data(index, Qt::UserRole).value<ModPlatform::IndexedPack>().addonId) {
|
|
||||||
// Cache info :^)
|
// Cache info :^)
|
||||||
QVariant new_pack;
|
QVariant new_pack;
|
||||||
new_pack.setValue(pack);
|
new_pack.setValue(current_pack);
|
||||||
if (!setData(index, new_pack, Qt::UserRole)) {
|
if (!setData(index, new_pack, Qt::UserRole)) {
|
||||||
qWarning() << "Failed to cache mod versions!";
|
qWarning() << "Failed to cache mod versions!";
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
emit versionListUpdated();
|
emit versionListUpdated();
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace ResourceDownload
|
} // namespace ResourceDownload
|
||||||
|
Loading…
x
Reference in New Issue
Block a user