diff options
author | Petr Mrázek <peterix@gmail.com> | 2017-09-26 13:38:34 +0200 |
---|---|---|
committer | Petr Mrázek <peterix@gmail.com> | 2017-09-26 13:38:34 +0200 |
commit | 2b998bb8cc8e83efafb20ed2fbe1a245d4d5ec5f (patch) | |
tree | 392872160002aa12c95a9f9239300aa12e80b368 /api | |
parent | eac892965e152519a6d7278504e7d4118aa7c01f (diff) | |
download | MultiMC-2b998bb8cc8e83efafb20ed2fbe1a245d4d5ec5f.tar MultiMC-2b998bb8cc8e83efafb20ed2fbe1a245d4d5ec5f.tar.gz MultiMC-2b998bb8cc8e83efafb20ed2fbe1a245d4d5ec5f.tar.lz MultiMC-2b998bb8cc8e83efafb20ed2fbe1a245d4d5ec5f.tar.xz MultiMC-2b998bb8cc8e83efafb20ed2fbe1a245d4d5ec5f.zip |
NOISSUE implement LEgacy instance migration
Diffstat (limited to 'api')
-rw-r--r-- | api/logic/CMakeLists.txt | 4 | ||||
-rw-r--r-- | api/logic/FolderInstanceProvider.cpp | 10 | ||||
-rw-r--r-- | api/logic/FolderInstanceProvider.h | 3 | ||||
-rw-r--r-- | api/logic/minecraft/legacy/LegacyMigrationTask.cpp | 51 | ||||
-rw-r--r-- | api/logic/minecraft/legacy/LegacyUpgradeTask.cpp | 133 | ||||
-rw-r--r-- | api/logic/minecraft/legacy/LegacyUpgradeTask.h (renamed from api/logic/minecraft/legacy/LegacyMigrationTask.h) | 5 |
6 files changed, 151 insertions, 55 deletions
diff --git a/api/logic/CMakeLists.txt b/api/logic/CMakeLists.txt index 0a65c6e2..a99d22ab 100644 --- a/api/logic/CMakeLists.txt +++ b/api/logic/CMakeLists.txt @@ -232,8 +232,8 @@ set(MINECRAFT_SOURCES minecraft/legacy/LegacyModList.cpp minecraft/legacy/LegacyInstance.h minecraft/legacy/LegacyInstance.cpp - minecraft/legacy/LegacyMigrationTask.h - minecraft/legacy/LegacyMigrationTask.cpp + minecraft/legacy/LegacyUpgradeTask.h + minecraft/legacy/LegacyUpgradeTask.cpp minecraft/GradleSpecifier.h minecraft/MinecraftInstance.cpp minecraft/MinecraftInstance.h diff --git a/api/logic/FolderInstanceProvider.cpp b/api/logic/FolderInstanceProvider.cpp index 25e9bb84..53618439 100644 --- a/api/logic/FolderInstanceProvider.cpp +++ b/api/logic/FolderInstanceProvider.cpp @@ -430,6 +430,16 @@ Task * FolderInstanceProvider::copyTask(const InstancePtr& oldInstance, const QS return new FolderInstanceStaging(this, task, stagingPath, instName, instGroup); } +// FIXME: find a better place for this +#include "minecraft/legacy/LegacyUpgradeTask.h" +Task * FolderInstanceProvider::legacyUpgradeTask(const InstancePtr& oldInstance) +{ + auto stagingPath = getStagedInstancePath(); + QString newName = tr("%1 (Migrated)").arg(oldInstance->name()); + auto task = new LegacyUpgradeTask(m_globalSettings, stagingPath, oldInstance, newName); + return new FolderInstanceStaging(this, task, stagingPath, newName, oldInstance->group()); +} + QString FolderInstanceProvider::getStagedInstancePath() { QString key = QUuid::createUuid().toString(); diff --git a/api/logic/FolderInstanceProvider.h b/api/logic/FolderInstanceProvider.h index b3c6d955..a8d1a7bf 100644 --- a/api/logic/FolderInstanceProvider.h +++ b/api/logic/FolderInstanceProvider.h @@ -28,6 +28,9 @@ public: // import zipped instance into this provider Task * zipImportTask(const QUrl sourceUrl, const QString &instName, const QString &instGroup, const QString &instIcon); + // migrate an instance to the current format + Task * legacyUpgradeTask(const InstancePtr& oldInstance); + /** * Create a new empty staging area for instance creation and @return a path/key top commit it later. * Used by instance manipulation tasks. diff --git a/api/logic/minecraft/legacy/LegacyMigrationTask.cpp b/api/logic/minecraft/legacy/LegacyMigrationTask.cpp deleted file mode 100644 index 0d3cef35..00000000 --- a/api/logic/minecraft/legacy/LegacyMigrationTask.cpp +++ /dev/null @@ -1,51 +0,0 @@ -#include "LegacyMigrationTask.h" -#include "BaseInstanceProvider.h" -#include "settings/INISettingsObject.h" -#include "FileSystem.h" -#include "NullInstance.h" -#include "pathmatcher/RegexpMatcher.h" -#include <QtConcurrentRun> - -LegacyMigrationTask::LegacyMigrationTask(SettingsObjectPtr settings, const QString & stagingPath, InstancePtr origInstance) -{ - m_globalSettings = settings; - m_stagingPath = stagingPath; - m_origInstance = origInstance; -} - -void LegacyMigrationTask::executeTask() -{ - setStatus(tr("Copying instance %1").arg(m_origInstance->name())); - - FS::copy folderCopy(m_origInstance->instanceRoot(), m_stagingPath); - folderCopy.followSymlinks(true); - - m_copyFuture = QtConcurrent::run(QThreadPool::globalInstance(), folderCopy); - connect(&m_copyFutureWatcher, &QFutureWatcher<bool>::finished, this, &LegacyMigrationTask::copyFinished); - connect(&m_copyFutureWatcher, &QFutureWatcher<bool>::canceled, this, &LegacyMigrationTask::copyAborted); - m_copyFutureWatcher.setFuture(m_copyFuture); -} - -void LegacyMigrationTask::copyFinished() -{ - auto successful = m_copyFuture.result(); - if(!successful) - { - emitFailed(tr("Instance folder copy failed.")); - return; - } - // FIXME: shouldn't this be able to report errors? - auto instanceSettings = std::make_shared<INISettingsObject>(FS::PathCombine(m_stagingPath, "instance.cfg")); - instanceSettings->registerSetting("InstanceType", "Legacy"); - - InstancePtr inst(new NullInstance(m_globalSettings, instanceSettings, m_stagingPath)); - inst->setName(tr("%1 (Migrated)").arg(m_origInstance->name())); - emitSucceeded(); -} - -void LegacyMigrationTask::copyAborted() -{ - emitFailed(tr("Instance folder copy has been aborted.")); - return; -} - diff --git a/api/logic/minecraft/legacy/LegacyUpgradeTask.cpp b/api/logic/minecraft/legacy/LegacyUpgradeTask.cpp new file mode 100644 index 00000000..cb0572ed --- /dev/null +++ b/api/logic/minecraft/legacy/LegacyUpgradeTask.cpp @@ -0,0 +1,133 @@ +#include "LegacyUpgradeTask.h" +#include "BaseInstanceProvider.h" +#include "settings/INISettingsObject.h" +#include "FileSystem.h" +#include "NullInstance.h" +#include "pathmatcher/RegexpMatcher.h" +#include <QtConcurrentRun> +#include "LegacyInstance.h" +#include "minecraft/MinecraftInstance.h" +#include "minecraft/MinecraftProfile.h" + +LegacyUpgradeTask::LegacyUpgradeTask(SettingsObjectPtr settings, const QString & stagingPath, InstancePtr origInstance, const QString & newName) +{ + m_globalSettings = settings; + m_stagingPath = stagingPath; + m_origInstance = origInstance; + m_newName = newName; +} + +void LegacyUpgradeTask::executeTask() +{ + setStatus(tr("Copying instance %1").arg(m_origInstance->name())); + + FS::copy folderCopy(m_origInstance->instanceRoot(), m_stagingPath); + folderCopy.followSymlinks(true); + + m_copyFuture = QtConcurrent::run(QThreadPool::globalInstance(), folderCopy); + connect(&m_copyFutureWatcher, &QFutureWatcher<bool>::finished, this, &LegacyUpgradeTask::copyFinished); + connect(&m_copyFutureWatcher, &QFutureWatcher<bool>::canceled, this, &LegacyUpgradeTask::copyAborted); + m_copyFutureWatcher.setFuture(m_copyFuture); +} + +static QString decideVersion(const QString& currentVersion, const QString& intendedVersion) +{ + if(intendedVersion != currentVersion) + { + if(!intendedVersion.isEmpty()) + { + return intendedVersion; + } + else if(!currentVersion.isEmpty()) + { + return currentVersion; + } + } + else + { + if(!intendedVersion.isEmpty()) + { + return intendedVersion; + } + } + // TODO: possibly add fallback to the old jar/classfile analysis method from MultiMC4 + return QString(); +} + +void LegacyUpgradeTask::copyFinished() +{ + auto successful = m_copyFuture.result(); + if(!successful) + { + emitFailed(tr("Instance folder copy failed.")); + return; + } + auto legacyInst = std::dynamic_pointer_cast<LegacyInstance>(m_origInstance); + + auto instanceSettings = std::make_shared<INISettingsObject>(FS::PathCombine(m_stagingPath, "instance.cfg")); + instanceSettings->registerSetting("InstanceType", "Legacy"); + instanceSettings->set("InstanceType", "OneSix"); + std::shared_ptr<MinecraftInstance> inst(new MinecraftInstance(m_globalSettings, instanceSettings, m_stagingPath)); + inst->setName(m_newName); + inst->init(); + + QString preferredVersionNumber = decideVersion(legacyInst->currentVersionId(), legacyInst->intendedVersionId()); + if(preferredVersionNumber.isNull()) + { + // FIXME: let the user decide + emitFailed(tr("Could not decide Minecraft version.")); + return; + } + inst->setComponentVersion("net.minecraft", preferredVersionNumber); + + // BUG: reloadProfile should not be necessary, but setComponentVersion voids the profile created by init()! + inst->reloadProfile(); + auto profile = inst->getMinecraftProfile(); + + if(legacyInst->shouldUseCustomBaseJar()) + { + QString jarPath = legacyInst->customBaseJar(); + qDebug() << "Base jar is custom! : " << jarPath; + // FIXME: handle case when the jar is unreadable? + // TODO: check the hash, if it's the same as the upstream jar, do not do this + profile->installCustomJar(jarPath); + } + + auto jarMods = legacyInst->getJarMods(); + for(auto & jarMod: jarMods) + { + QString modPath = jarMod.filename().absoluteFilePath(); + qDebug() << "jarMod: " << modPath; + profile->installJarMods({modPath}); + } + + // remove all the extra garbage we no longer need + auto removeAll = [&](const QString &root, const QStringList &things) + { + for(auto &thing : things) + { + auto removePath = FS::PathCombine(root, thing); + QFileInfo stat(removePath); + if(stat.isDir()) + { + FS::deletePath(removePath); + } + else + { + QFile::remove(removePath); + } + } + }; + QStringList rootRemovables = {"modlist", "version", "instMods"}; + QStringList mcRemovables = {"bin", "MultiMCLauncher.jar", "icon.png"}; + removeAll(inst->instanceRoot(), rootRemovables); + removeAll(inst->minecraftRoot(), mcRemovables); + emitSucceeded(); +} + +void LegacyUpgradeTask::copyAborted() +{ + emitFailed(tr("Instance folder copy has been aborted.")); + return; +} + diff --git a/api/logic/minecraft/legacy/LegacyMigrationTask.h b/api/logic/minecraft/legacy/LegacyUpgradeTask.h index 36cfa240..56896385 100644 --- a/api/logic/minecraft/legacy/LegacyMigrationTask.h +++ b/api/logic/minecraft/legacy/LegacyUpgradeTask.h @@ -13,11 +13,11 @@ class BaseInstanceProvider; -class MULTIMC_LOGIC_EXPORT LegacyMigrationTask : public Task +class MULTIMC_LOGIC_EXPORT LegacyUpgradeTask : public Task { Q_OBJECT public: - explicit LegacyMigrationTask(SettingsObjectPtr settings, const QString & stagingPath, InstancePtr origInstance); + explicit LegacyUpgradeTask(SettingsObjectPtr settings, const QString & stagingPath, InstancePtr origInstance, const QString & newName); protected: //! Entry point for tasks. @@ -29,6 +29,7 @@ private: /* data */ SettingsObjectPtr m_globalSettings; InstancePtr m_origInstance; QString m_stagingPath; + QString m_newName; QFuture<bool> m_copyFuture; QFutureWatcher<bool> m_copyFutureWatcher; }; |