2021-01-18 08:28:54 +01:00
|
|
|
/* Copyright 2015-2021 MultiMC Contributors
|
2016-04-06 23:09:30 +02:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2017-03-11 01:39:45 +01:00
|
|
|
#include "BaseEntity.h"
|
2016-04-06 23:09:30 +02:00
|
|
|
|
2023-08-14 18:16:53 +02:00
|
|
|
#include "Json.h"
|
2023-06-01 16:39:04 -07:00
|
|
|
#include "net/ApiDownload.h"
|
2017-03-18 02:22:36 +01:00
|
|
|
#include "net/HttpMetaCache.h"
|
|
|
|
#include "net/NetJob.h"
|
|
|
|
|
2021-11-21 23:21:12 +01:00
|
|
|
#include "Application.h"
|
2023-08-14 18:16:53 +02:00
|
|
|
#include "BuildConfig.h"
|
2020-07-18 16:18:02 +02:00
|
|
|
|
2023-08-14 18:16:53 +02:00
|
|
|
class ParsingValidator : public Net::Validator {
|
|
|
|
public: /* con/des */
|
|
|
|
ParsingValidator(Meta::BaseEntity* entity) : m_entity(entity){};
|
|
|
|
virtual ~ParsingValidator(){};
|
2017-03-18 02:22:36 +01:00
|
|
|
|
2023-08-14 18:16:53 +02:00
|
|
|
public: /* methods */
|
|
|
|
bool init(QNetworkRequest&) override { return true; }
|
|
|
|
bool write(QByteArray& data) override
|
2017-03-18 02:22:36 +01:00
|
|
|
{
|
2023-06-30 23:51:15 -07:00
|
|
|
this->m_data.append(data);
|
2017-03-18 02:22:36 +01:00
|
|
|
return true;
|
|
|
|
}
|
2023-08-14 18:16:53 +02:00
|
|
|
bool abort() override { return true; }
|
|
|
|
bool validate(QNetworkReply&) override
|
2017-03-18 02:22:36 +01:00
|
|
|
{
|
|
|
|
auto fname = m_entity->localFilename();
|
2023-08-14 18:16:53 +02:00
|
|
|
try {
|
2023-06-30 23:51:15 -07:00
|
|
|
auto doc = Json::requireDocument(m_data, fname);
|
2020-06-28 13:25:53 +02:00
|
|
|
auto obj = Json::requireObject(doc, fname);
|
|
|
|
m_entity->parse(obj);
|
2017-03-18 02:22:36 +01:00
|
|
|
return true;
|
2023-08-14 18:16:53 +02:00
|
|
|
} catch (const Exception& e) {
|
2017-03-18 02:22:36 +01:00
|
|
|
qWarning() << "Unable to parse response:" << e.cause();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-14 18:16:53 +02:00
|
|
|
private: /* data */
|
2023-06-30 23:51:15 -07:00
|
|
|
QByteArray m_data;
|
2023-08-14 18:16:53 +02:00
|
|
|
Meta::BaseEntity* m_entity;
|
2017-03-18 02:22:36 +01:00
|
|
|
};
|
|
|
|
|
2023-08-14 18:16:53 +02:00
|
|
|
Meta::BaseEntity::~BaseEntity() {}
|
2016-04-06 23:09:30 +02:00
|
|
|
|
2017-03-18 02:22:36 +01:00
|
|
|
QUrl Meta::BaseEntity::url() const
|
2016-04-06 23:09:30 +02:00
|
|
|
{
|
2022-04-13 23:00:32 +02:00
|
|
|
auto s = APPLICATION->settings();
|
|
|
|
QString metaOverride = s->get("MetaURLOverride").toString();
|
2023-08-14 18:16:53 +02:00
|
|
|
if (metaOverride.isEmpty()) {
|
2022-04-13 23:00:32 +02:00
|
|
|
return QUrl(BuildConfig.META_URL).resolved(localFilename());
|
2023-08-14 18:16:53 +02:00
|
|
|
} else {
|
2022-04-13 23:00:32 +02:00
|
|
|
return QUrl(metaOverride).resolved(localFilename());
|
|
|
|
}
|
2016-04-06 23:09:30 +02:00
|
|
|
}
|
|
|
|
|
2017-03-18 02:22:36 +01:00
|
|
|
bool Meta::BaseEntity::loadLocalFile()
|
2016-04-06 23:09:30 +02:00
|
|
|
{
|
2017-03-18 02:22:36 +01:00
|
|
|
const QString fname = QDir("meta").absoluteFilePath(localFilename());
|
2023-08-14 18:16:53 +02:00
|
|
|
if (!QFile::exists(fname)) {
|
2017-03-18 02:22:36 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// TODO: check if the file has the expected checksum
|
2023-08-14 18:16:53 +02:00
|
|
|
try {
|
2020-06-28 13:25:53 +02:00
|
|
|
auto doc = Json::requireDocument(fname, fname);
|
|
|
|
auto obj = Json::requireObject(doc, fname);
|
|
|
|
parse(obj);
|
2017-03-18 02:22:36 +01:00
|
|
|
return true;
|
2023-08-14 18:16:53 +02:00
|
|
|
} catch (const Exception& e) {
|
2017-03-18 02:22:36 +01:00
|
|
|
qDebug() << QString("Unable to parse file %1: %2").arg(fname, e.cause());
|
|
|
|
// just make sure it's gone and we never consider it again.
|
|
|
|
QFile::remove(fname);
|
|
|
|
return false;
|
|
|
|
}
|
2016-04-06 23:09:30 +02:00
|
|
|
}
|
2017-03-17 01:48:54 +01:00
|
|
|
|
2017-11-11 01:38:31 +01:00
|
|
|
void Meta::BaseEntity::load(Net::Mode loadType)
|
2016-04-06 23:09:30 +02:00
|
|
|
{
|
2017-03-19 23:58:54 +01:00
|
|
|
// load local file if nothing is loaded yet
|
2023-08-14 18:16:53 +02:00
|
|
|
if (!isLoaded()) {
|
|
|
|
if (loadLocalFile()) {
|
2017-03-19 23:58:54 +01:00
|
|
|
m_loadStatus = LoadStatus::Local;
|
|
|
|
}
|
2017-03-18 02:22:36 +01:00
|
|
|
}
|
2017-03-19 23:58:54 +01:00
|
|
|
// if we need remote update, run the update task
|
2023-08-14 18:16:53 +02:00
|
|
|
if (loadType == Net::Mode::Offline || !shouldStartRemoteUpdate()) {
|
2017-03-18 02:22:36 +01:00
|
|
|
return;
|
|
|
|
}
|
2023-01-24 16:52:09 -03:00
|
|
|
m_updateTask.reset(new NetJob(QObject::tr("Download of meta file %1").arg(localFilename()), APPLICATION->network()));
|
2017-03-18 02:22:36 +01:00
|
|
|
auto url = this->url();
|
2021-11-21 23:21:12 +01:00
|
|
|
auto entry = APPLICATION->metacache()->resolveEntry("meta", localFilename());
|
2017-03-18 02:22:36 +01:00
|
|
|
entry->setStale(true);
|
2023-06-01 16:39:04 -07:00
|
|
|
auto dl = Net::ApiDownload::makeCached(url, entry);
|
2017-03-18 02:22:36 +01:00
|
|
|
/*
|
|
|
|
* The validator parses the file and loads it into the object.
|
|
|
|
* If that fails, the file is not written to storage.
|
|
|
|
*/
|
|
|
|
dl->addValidator(new ParsingValidator(this));
|
2021-11-21 23:21:12 +01:00
|
|
|
m_updateTask->addNetAction(dl);
|
2017-03-18 02:22:36 +01:00
|
|
|
m_updateStatus = UpdateStatus::InProgress;
|
2023-08-14 18:16:53 +02:00
|
|
|
QObject::connect(m_updateTask.get(), &NetJob::succeeded, [&]() {
|
2017-03-18 02:22:36 +01:00
|
|
|
m_loadStatus = LoadStatus::Remote;
|
|
|
|
m_updateStatus = UpdateStatus::Succeeded;
|
|
|
|
m_updateTask.reset();
|
|
|
|
});
|
2023-08-14 18:16:53 +02:00
|
|
|
QObject::connect(m_updateTask.get(), &NetJob::failed, [&]() {
|
2017-03-18 02:22:36 +01:00
|
|
|
m_updateStatus = UpdateStatus::Failed;
|
|
|
|
m_updateTask.reset();
|
|
|
|
});
|
2021-12-31 05:27:59 +01:00
|
|
|
m_updateTask->start();
|
2017-03-11 01:39:45 +01:00
|
|
|
}
|
2017-03-18 02:22:36 +01:00
|
|
|
|
2017-04-24 01:30:51 +02:00
|
|
|
bool Meta::BaseEntity::isLoaded() const
|
|
|
|
{
|
|
|
|
return m_loadStatus > LoadStatus::NotLoaded;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Meta::BaseEntity::shouldStartRemoteUpdate() const
|
|
|
|
{
|
|
|
|
// TODO: version-locks and offline mode?
|
|
|
|
return m_updateStatus != UpdateStatus::InProgress;
|
|
|
|
}
|
|
|
|
|
2021-11-21 23:36:55 +01:00
|
|
|
Task::Ptr Meta::BaseEntity::getCurrentTask()
|
2017-03-19 02:13:49 +01:00
|
|
|
{
|
2023-08-14 18:16:53 +02:00
|
|
|
if (m_updateStatus == UpdateStatus::InProgress) {
|
2017-03-19 02:13:49 +01:00
|
|
|
return m_updateTask;
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|