Runnable 1.6 instances!
This commit is contained in:
parent
005a010ee6
commit
183a735145
@ -69,8 +69,23 @@ public:
|
|||||||
QString group() const;
|
QString group() const;
|
||||||
void setGroup(QString val);
|
void setGroup(QString val);
|
||||||
|
|
||||||
|
virtual QString intendedVersionId() const = 0;
|
||||||
virtual bool setIntendedVersionId(QString version) = 0;
|
virtual bool setIntendedVersionId(QString version) = 0;
|
||||||
virtual QString intendedVersionId() = 0;
|
|
||||||
|
/*!
|
||||||
|
* The instance's current version.
|
||||||
|
* This value represents the instance's current version. If this value is
|
||||||
|
* different from the intendedVersion, the instance should be updated.
|
||||||
|
* \warning Don't change this value unless you know what you're doing.
|
||||||
|
*/
|
||||||
|
virtual QString currentVersionId() const = 0;
|
||||||
|
//virtual void setCurrentVersionId(QString val) = 0;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Whether or not Minecraft should be downloaded when the instance is launched.
|
||||||
|
*/
|
||||||
|
virtual bool shouldUpdate() const = 0;
|
||||||
|
virtual void setShouldUpdate(bool val) = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the time that the instance was last launched.
|
* Gets the time that the instance was last launched.
|
||||||
@ -107,6 +122,8 @@ public:
|
|||||||
/// returns a valid minecraft process, ready for launch
|
/// returns a valid minecraft process, ready for launch
|
||||||
virtual MinecraftProcess* prepareForLaunch(QString user, QString session) = 0;
|
virtual MinecraftProcess* prepareForLaunch(QString user, QString session) = 0;
|
||||||
|
|
||||||
|
/// do any necessary cleanups after the instance finishes. also runs before 'prepareForLaunch'
|
||||||
|
virtual void cleanupAfterRun() = 0;
|
||||||
signals:
|
signals:
|
||||||
/*!
|
/*!
|
||||||
* \brief Signal emitted when properties relevant to the instance view change
|
* \brief Signal emitted when properties relevant to the instance view change
|
||||||
|
@ -91,6 +91,6 @@ add_definitions(-DLIBMULTIMC_LIBRARY)
|
|||||||
|
|
||||||
add_library(backend SHARED ${LIBINST_SOURCES} ${LIBINST_HEADERS})
|
add_library(backend SHARED ${LIBINST_SOURCES} ${LIBINST_HEADERS})
|
||||||
qt5_use_modules(backend Core Network Xml)
|
qt5_use_modules(backend Core Network Xml)
|
||||||
target_link_libraries(backend libUtil libSettings)
|
target_link_libraries(backend libUtil libSettings quazip)
|
||||||
|
|
||||||
|
|
||||||
|
@ -16,7 +16,7 @@ LegacyInstance::LegacyInstance(const QString& rootDir, SettingsObject* settings,
|
|||||||
settings->registerSetting(new Setting("NeedsRebuild", true));
|
settings->registerSetting(new Setting("NeedsRebuild", true));
|
||||||
settings->registerSetting(new Setting("ShouldUpdate", false));
|
settings->registerSetting(new Setting("ShouldUpdate", false));
|
||||||
settings->registerSetting(new Setting("JarVersion", "Unknown"));
|
settings->registerSetting(new Setting("JarVersion", "Unknown"));
|
||||||
settings->registerSetting(new Setting("LwjglVersion", "2.9.0"));
|
settings->registerSetting(new Setting("LwjglVersion", "Mojang"));
|
||||||
settings->registerSetting(new Setting("IntendedJarVersion", ""));
|
settings->registerSetting(new Setting("IntendedJarVersion", ""));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -93,6 +93,11 @@ MinecraftProcess* LegacyInstance::prepareForLaunch(QString user, QString session
|
|||||||
return proc;
|
return proc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void LegacyInstance::cleanupAfterRun()
|
||||||
|
{
|
||||||
|
//FIXME: delete the launcher and icons and whatnot.
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
QString LegacyInstance::instModsDir() const
|
QString LegacyInstance::instModsDir() const
|
||||||
{
|
{
|
||||||
@ -152,7 +157,7 @@ void LegacyInstance::updateCurrentVersion(bool keepCurrent)
|
|||||||
if(!jar.exists())
|
if(!jar.exists())
|
||||||
{
|
{
|
||||||
setLastCurrentVersionUpdate(0);
|
setLastCurrentVersionUpdate(0);
|
||||||
setCurrentVersion("Unknown");
|
setCurrentVersionId("Unknown");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -163,7 +168,7 @@ void LegacyInstance::updateCurrentVersion(bool keepCurrent)
|
|||||||
{
|
{
|
||||||
// TODO: Implement GetMinecraftJarVersion function.
|
// TODO: Implement GetMinecraftJarVersion function.
|
||||||
QString newVersion = "Unknown";//javautils::GetMinecraftJarVersion(jar.absoluteFilePath());
|
QString newVersion = "Unknown";//javautils::GetMinecraftJarVersion(jar.absoluteFilePath());
|
||||||
setCurrentVersion(newVersion);
|
setCurrentVersionId(newVersion);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
qint64 LegacyInstance::lastCurrentVersionUpdate() const
|
qint64 LegacyInstance::lastCurrentVersionUpdate() const
|
||||||
@ -186,16 +191,18 @@ void LegacyInstance::setShouldRebuild ( bool val )
|
|||||||
I_D(LegacyInstance);
|
I_D(LegacyInstance);
|
||||||
d->m_settings->set ( "NeedsRebuild", val );
|
d->m_settings->set ( "NeedsRebuild", val );
|
||||||
}
|
}
|
||||||
QString LegacyInstance::currentVersion() const
|
QString LegacyInstance::currentVersionId() const
|
||||||
{
|
{
|
||||||
I_D(LegacyInstance);
|
I_D(LegacyInstance);
|
||||||
return d->m_settings->get ( "JarVersion" ).toString();
|
return d->m_settings->get ( "JarVersion" ).toString();
|
||||||
}
|
}
|
||||||
void LegacyInstance::setCurrentVersion ( QString val )
|
|
||||||
|
void LegacyInstance::setCurrentVersionId ( QString val )
|
||||||
{
|
{
|
||||||
I_D(LegacyInstance);
|
I_D(LegacyInstance);
|
||||||
d->m_settings->set ( "JarVersion", val );
|
d->m_settings->set ( "JarVersion", val );
|
||||||
}
|
}
|
||||||
|
|
||||||
QString LegacyInstance::lwjglVersion() const
|
QString LegacyInstance::lwjglVersion() const
|
||||||
{
|
{
|
||||||
I_D(LegacyInstance);
|
I_D(LegacyInstance);
|
||||||
@ -206,36 +213,17 @@ void LegacyInstance::setLWJGLVersion ( QString val )
|
|||||||
I_D(LegacyInstance);
|
I_D(LegacyInstance);
|
||||||
d->m_settings->set ( "LwjglVersion", val );
|
d->m_settings->set ( "LwjglVersion", val );
|
||||||
}
|
}
|
||||||
QString LegacyInstance::intendedVersionId()
|
QString LegacyInstance::intendedVersionId() const
|
||||||
{
|
{
|
||||||
I_D(LegacyInstance);
|
I_D(LegacyInstance);
|
||||||
return d->m_settings->get ( "IntendedJarVersion" ).toString();
|
return d->m_settings->get ( "IntendedJarVersion" ).toString();
|
||||||
}
|
}
|
||||||
bool LegacyInstance::setIntendedVersionId ( QString version )
|
bool LegacyInstance::setIntendedVersionId ( QString version )
|
||||||
{
|
{
|
||||||
/*
|
|
||||||
I_D(LegacyInstance);
|
|
||||||
d->m_settings->set ( "IntendedJarVersion", val );
|
|
||||||
*/
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
bool LegacyInstance::shouldUpdate() const
|
bool LegacyInstance::shouldUpdate() const
|
||||||
{
|
{
|
||||||
/*
|
|
||||||
I_D(LegacyInstance);
|
|
||||||
QVariant var = d->m_settings->get ( "ShouldUpdate" );
|
|
||||||
if ( !var.isValid() || var.toBool() == false )
|
|
||||||
{
|
|
||||||
return intendedVersionId() != currentVersion();
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
*/
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
void LegacyInstance::setShouldUpdate ( bool val )
|
void LegacyInstance::setShouldUpdate ( bool val ) {}
|
||||||
{
|
|
||||||
/*
|
|
||||||
I_D(LegacyInstance);
|
|
||||||
d->m_settings->set ( "ShouldUpdate", val );
|
|
||||||
*/
|
|
||||||
}
|
|
||||||
|
@ -56,7 +56,6 @@ public:
|
|||||||
qint64 lastCurrentVersionUpdate() const;
|
qint64 lastCurrentVersionUpdate() const;
|
||||||
void setLastCurrentVersionUpdate(qint64 val);
|
void setLastCurrentVersionUpdate(qint64 val);
|
||||||
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Whether or not the instance's minecraft.jar needs to be rebuilt.
|
* Whether or not the instance's minecraft.jar needs to be rebuilt.
|
||||||
* If this is true, when the instance launches, its jar mods will be
|
* If this is true, when the instance launches, its jar mods will be
|
||||||
@ -65,39 +64,21 @@ public:
|
|||||||
bool shouldRebuild() const;
|
bool shouldRebuild() const;
|
||||||
void setShouldRebuild(bool val);
|
void setShouldRebuild(bool val);
|
||||||
|
|
||||||
|
virtual QString currentVersionId() const;
|
||||||
|
virtual void setCurrentVersionId(QString val);
|
||||||
|
|
||||||
/*!
|
|
||||||
* The instance's current version.
|
|
||||||
* This value represents the instance's current version. If this value is
|
|
||||||
* different from the intendedVersion, the instance should be updated.
|
|
||||||
* \warning Don't change this value unless you know what you're doing.
|
|
||||||
*/
|
|
||||||
QString currentVersion() const;
|
|
||||||
void setCurrentVersion(QString val);
|
|
||||||
|
|
||||||
//! The version of LWJGL that this instance uses.
|
//! The version of LWJGL that this instance uses.
|
||||||
QString lwjglVersion() const;
|
QString lwjglVersion() const;
|
||||||
|
/// st the version of LWJGL libs this instance will use
|
||||||
void setLWJGLVersion(QString val);
|
void setLWJGLVersion(QString val);
|
||||||
|
|
||||||
/*!
|
virtual QString intendedVersionId() const;
|
||||||
* The version that the user has set for this instance to use.
|
|
||||||
* If this is not the same as currentVersion, the instance's game updater
|
|
||||||
* will be run on launch.
|
|
||||||
*/
|
|
||||||
virtual QString intendedVersionId();
|
|
||||||
virtual bool setIntendedVersionId ( QString version );
|
virtual bool setIntendedVersionId ( QString version );
|
||||||
|
|
||||||
/*!
|
virtual bool shouldUpdate() const;
|
||||||
* Whether or not Minecraft should be downloaded when the instance is launched.
|
virtual void setShouldUpdate(bool val);
|
||||||
* This returns true if shouldForceUpdate game is true or if the intended and
|
|
||||||
* current versions don't match.
|
|
||||||
*/
|
|
||||||
bool shouldUpdate() const;
|
|
||||||
void setShouldUpdate(bool val);
|
|
||||||
|
|
||||||
/// return a valid GameUpdateTask if an update is needed, return NULL otherwise
|
|
||||||
virtual OneSixUpdate* doUpdate();
|
virtual OneSixUpdate* doUpdate();
|
||||||
|
|
||||||
/// prepare the instance for launch and return a constructed MinecraftProcess instance
|
|
||||||
virtual MinecraftProcess* prepareForLaunch( QString user, QString session );
|
virtual MinecraftProcess* prepareForLaunch( QString user, QString session );
|
||||||
|
virtual void cleanupAfterRun();
|
||||||
};
|
};
|
@ -134,6 +134,7 @@ void MinecraftProcess::finish(int code, ExitStatus status)
|
|||||||
//TODO: error handling
|
//TODO: error handling
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
m_instance->cleanupAfterRun();
|
||||||
emit ended();
|
emit ended();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2,13 +2,20 @@
|
|||||||
#include "OneSixInstance_p.h"
|
#include "OneSixInstance_p.h"
|
||||||
#include "OneSixUpdate.h"
|
#include "OneSixUpdate.h"
|
||||||
#include "MinecraftProcess.h"
|
#include "MinecraftProcess.h"
|
||||||
|
#include "VersionFactory.h"
|
||||||
|
|
||||||
#include <setting.h>
|
#include <setting.h>
|
||||||
|
#include <pathutils.h>
|
||||||
|
#include <cmdutils.h>
|
||||||
|
#include <JlCompress.h>
|
||||||
|
|
||||||
OneSixInstance::OneSixInstance ( const QString& rootDir, SettingsObject* setting_obj, QObject* parent )
|
OneSixInstance::OneSixInstance ( const QString& rootDir, SettingsObject* setting_obj, QObject* parent )
|
||||||
: BaseInstance ( new OneSixInstancePrivate(), rootDir, setting_obj, parent )
|
: BaseInstance ( new OneSixInstancePrivate(), rootDir, setting_obj, parent )
|
||||||
{
|
{
|
||||||
I_D(OneSixInstance);
|
I_D(OneSixInstance);
|
||||||
d->m_settings->registerSetting(new Setting("IntendedVersion", ""));
|
d->m_settings->registerSetting(new Setting("IntendedVersion", ""));
|
||||||
|
d->m_settings->registerSetting(new Setting("ShouldUpdate", false));
|
||||||
|
reloadFullVersion();
|
||||||
}
|
}
|
||||||
|
|
||||||
OneSixUpdate* OneSixInstance::doUpdate()
|
OneSixUpdate* OneSixInstance::doUpdate()
|
||||||
@ -16,17 +23,188 @@ OneSixUpdate* OneSixInstance::doUpdate()
|
|||||||
return new OneSixUpdate(this);
|
return new OneSixUpdate(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QString replaceTokensIn(QString text, QMap<QString, QString> with)
|
||||||
|
{
|
||||||
|
QString result;
|
||||||
|
QRegExp token_regexp("\\$\\{(.+)\\}");
|
||||||
|
token_regexp.setMinimal(true);
|
||||||
|
QStringList list;
|
||||||
|
int tail = 0;
|
||||||
|
int head = 0;
|
||||||
|
while ((head = token_regexp.indexIn(text, head)) != -1)
|
||||||
|
{
|
||||||
|
result.append(text.mid(tail, head-tail));
|
||||||
|
QString key = token_regexp.cap(1);
|
||||||
|
auto iter = with.find(key);
|
||||||
|
if(iter != with.end())
|
||||||
|
{
|
||||||
|
result.append(*iter);
|
||||||
|
}
|
||||||
|
head += token_regexp.matchedLength();
|
||||||
|
tail = head;
|
||||||
|
}
|
||||||
|
result.append(text.mid(tail));
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
QStringList OneSixInstance::processMinecraftArgs( QString user, QString session )
|
||||||
|
{
|
||||||
|
I_D(OneSixInstance);
|
||||||
|
auto version = d->version;
|
||||||
|
QString args_pattern = version->minecraftArguments;
|
||||||
|
|
||||||
|
QMap<QString, QString> token_mapping;
|
||||||
|
token_mapping["auth_username"] = user;
|
||||||
|
token_mapping["auth_session"] = session;
|
||||||
|
//FIXME: user and player name are DIFFERENT!
|
||||||
|
token_mapping["auth_player_name"] = user;
|
||||||
|
//FIXME: WTF is this. I just plugged in a random UUID here.
|
||||||
|
token_mapping["auth_uuid"] = "7d4bacf0-fd62-11e2-b778-0800200c9a66"; // obviously fake.
|
||||||
|
|
||||||
|
// this is for offline:
|
||||||
|
/*
|
||||||
|
map["auth_player_name"] = "Player";
|
||||||
|
map["auth_player_name"] = "00000000-0000-0000-0000-000000000000";
|
||||||
|
*/
|
||||||
|
|
||||||
|
token_mapping["profile_name"] = name();
|
||||||
|
token_mapping["version_name"] = version->id;
|
||||||
|
|
||||||
|
QString absRootDir = QDir(rootDir()).absolutePath();
|
||||||
|
token_mapping["game_directory"] = absRootDir;
|
||||||
|
QString absAssetsDir = QDir("assets/").absolutePath();
|
||||||
|
token_mapping["game_assets"] = absAssetsDir;
|
||||||
|
|
||||||
|
QStringList parts = args_pattern.split(' ',QString::SkipEmptyParts);
|
||||||
|
for (int i = 0; i < parts.length(); i++)
|
||||||
|
{
|
||||||
|
parts[i] = replaceTokensIn(parts[i], token_mapping);
|
||||||
|
}
|
||||||
|
return parts;
|
||||||
|
}
|
||||||
|
|
||||||
MinecraftProcess* OneSixInstance::prepareForLaunch ( QString user, QString session )
|
MinecraftProcess* OneSixInstance::prepareForLaunch ( QString user, QString session )
|
||||||
{
|
{
|
||||||
return nullptr;
|
I_D(OneSixInstance);
|
||||||
|
cleanupAfterRun();
|
||||||
|
auto version = d->version;
|
||||||
|
if(!version)
|
||||||
|
return nullptr;
|
||||||
|
auto libs_to_extract = version->getActiveNativeLibs();
|
||||||
|
QString natives_dir_raw = PathCombine(rootDir(), "natives/");
|
||||||
|
bool success = ensurePathExists(natives_dir_raw);
|
||||||
|
if(!success)
|
||||||
|
{
|
||||||
|
// FIXME: handle errors
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
for(auto lib: libs_to_extract)
|
||||||
|
{
|
||||||
|
QString path = "libraries/" + lib->storagePath();
|
||||||
|
qDebug() << "Will extract " << path.toLocal8Bit();
|
||||||
|
if(JlCompress::extractWithExceptions(path, natives_dir_raw, lib->extract_excludes).isEmpty())
|
||||||
|
{
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QStringList args;
|
||||||
|
args.append(Util::Commandline::splitArgs(settings().get("JvmArgs").toString()));
|
||||||
|
args << QString("-Xms%1m").arg(settings().get("MinMemAlloc").toInt());
|
||||||
|
args << QString("-Xmx%1m").arg(settings().get("MaxMemAlloc").toInt());
|
||||||
|
QDir natives_dir(natives_dir_raw);
|
||||||
|
args << QString("-Djava.library.path=%1").arg( natives_dir.absolutePath() );
|
||||||
|
QString classPath;
|
||||||
|
{
|
||||||
|
auto libs = version->getActiveNormalLibs();
|
||||||
|
for (auto lib: libs)
|
||||||
|
{
|
||||||
|
QFileInfo fi(QString("libraries/") + lib->storagePath());
|
||||||
|
classPath.append(fi.absoluteFilePath());
|
||||||
|
//FIXME: make separator tweakable
|
||||||
|
classPath.append(':');
|
||||||
|
}
|
||||||
|
QString targetstr = "versions/" + version->id + "/" + version->id + ".jar";
|
||||||
|
QFileInfo fi(targetstr);
|
||||||
|
classPath.append(fi.absoluteFilePath());
|
||||||
|
}
|
||||||
|
if(classPath.size())
|
||||||
|
{
|
||||||
|
args << "-cp";
|
||||||
|
args << classPath;
|
||||||
|
}
|
||||||
|
args << version->mainClass;
|
||||||
|
args.append(processMinecraftArgs(user, session));
|
||||||
|
|
||||||
|
// create the process and set its parameters
|
||||||
|
MinecraftProcess * proc = new MinecraftProcess(this);
|
||||||
|
proc->setMinecraftArguments(args);
|
||||||
|
proc->setMinecraftWorkdir(rootDir());
|
||||||
|
return proc;
|
||||||
|
}
|
||||||
|
|
||||||
|
void OneSixInstance::cleanupAfterRun()
|
||||||
|
{
|
||||||
|
QString target_dir = PathCombine(rootDir(), "natives/");
|
||||||
|
QDir dir(target_dir);
|
||||||
|
dir.removeRecursively();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OneSixInstance::setIntendedVersionId ( QString version )
|
bool OneSixInstance::setIntendedVersionId ( QString version )
|
||||||
{
|
{
|
||||||
settings().set("IntendedVersion", version);
|
settings().set("IntendedVersion", version);
|
||||||
|
setShouldUpdate(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
QString OneSixInstance::intendedVersionId()
|
QString OneSixInstance::intendedVersionId() const
|
||||||
{
|
{
|
||||||
return settings().get("IntendedVersion").toString();
|
return settings().get("IntendedVersion").toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void OneSixInstance::setShouldUpdate ( bool val )
|
||||||
|
{
|
||||||
|
settings().set ( "ShouldUpdate", val );
|
||||||
|
}
|
||||||
|
|
||||||
|
bool OneSixInstance::shouldUpdate() const
|
||||||
|
{
|
||||||
|
I_D(OneSixInstance);
|
||||||
|
QVariant var = settings().get ( "ShouldUpdate" );
|
||||||
|
if ( !var.isValid() || var.toBool() == false )
|
||||||
|
{
|
||||||
|
return intendedVersionId() != currentVersionId();
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString OneSixInstance::currentVersionId() const
|
||||||
|
{
|
||||||
|
return intendedVersionId();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool OneSixInstance::reloadFullVersion()
|
||||||
|
{
|
||||||
|
I_D(OneSixInstance);
|
||||||
|
|
||||||
|
QString verpath = PathCombine(rootDir(), "version.json");
|
||||||
|
QFile versionfile(verpath);
|
||||||
|
if(versionfile.exists() && versionfile.open(QIODevice::ReadOnly))
|
||||||
|
{
|
||||||
|
FullVersionFactory fvf;
|
||||||
|
auto version = fvf.parse(versionfile.readAll());
|
||||||
|
versionfile.close();
|
||||||
|
if(version)
|
||||||
|
{
|
||||||
|
d->version = version;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
QSharedPointer< FullVersion > OneSixInstance::getFullVersion()
|
||||||
|
{
|
||||||
|
I_D(OneSixInstance);
|
||||||
|
return d->version;
|
||||||
|
}
|
||||||
|
@ -1,6 +1,9 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "BaseInstance.h"
|
#include "BaseInstance.h"
|
||||||
|
#include <QStringList>
|
||||||
|
class FullVersion;
|
||||||
|
|
||||||
class LIBMULTIMC_EXPORT OneSixInstance : public BaseInstance
|
class LIBMULTIMC_EXPORT OneSixInstance : public BaseInstance
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
@ -8,8 +11,21 @@ public:
|
|||||||
explicit OneSixInstance(const QString &rootDir, SettingsObject * settings, QObject *parent = 0);
|
explicit OneSixInstance(const QString &rootDir, SettingsObject * settings, QObject *parent = 0);
|
||||||
virtual OneSixUpdate* doUpdate();
|
virtual OneSixUpdate* doUpdate();
|
||||||
virtual MinecraftProcess* prepareForLaunch ( QString user, QString session );
|
virtual MinecraftProcess* prepareForLaunch ( QString user, QString session );
|
||||||
|
virtual void cleanupAfterRun();
|
||||||
|
|
||||||
|
virtual QString intendedVersionId() const;
|
||||||
virtual bool setIntendedVersionId ( QString version );
|
virtual bool setIntendedVersionId ( QString version );
|
||||||
virtual QString intendedVersionId();
|
|
||||||
|
|
||||||
|
virtual QString currentVersionId() const;
|
||||||
|
// virtual void setCurrentVersionId ( QString val ) {};
|
||||||
|
|
||||||
|
virtual bool shouldUpdate() const;
|
||||||
|
virtual void setShouldUpdate(bool val);
|
||||||
|
|
||||||
|
/// reload the full version json file. return true on success!
|
||||||
|
bool reloadFullVersion();
|
||||||
|
/// get the current full version info
|
||||||
|
QSharedPointer<FullVersion> getFullVersion();
|
||||||
|
private:
|
||||||
|
QStringList processMinecraftArgs( QString user, QString session );
|
||||||
};
|
};
|
@ -1,8 +1,9 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <QString>
|
|
||||||
#include <settingsobject.h>
|
|
||||||
#include "BaseInstance_p.h"
|
#include "BaseInstance_p.h"
|
||||||
|
#include "OneSixVersion.h"
|
||||||
|
|
||||||
struct OneSixInstancePrivate: public BaseInstancePrivate
|
struct OneSixInstancePrivate: public BaseInstancePrivate
|
||||||
{
|
{
|
||||||
|
QSharedPointer<FullVersion> version;
|
||||||
};
|
};
|
@ -28,6 +28,7 @@
|
|||||||
#include "lists/MinecraftVersionList.h"
|
#include "lists/MinecraftVersionList.h"
|
||||||
#include "VersionFactory.h"
|
#include "VersionFactory.h"
|
||||||
#include "OneSixVersion.h"
|
#include "OneSixVersion.h"
|
||||||
|
#include "OneSixInstance.h"
|
||||||
|
|
||||||
#include "pathutils.h"
|
#include "pathutils.h"
|
||||||
|
|
||||||
@ -40,14 +41,28 @@ OneSixUpdate::OneSixUpdate(BaseInstance *inst, QObject *parent) :
|
|||||||
|
|
||||||
void OneSixUpdate::executeTask()
|
void OneSixUpdate::executeTask()
|
||||||
{
|
{
|
||||||
|
QString intendedVersion = m_inst->intendedVersionId();
|
||||||
// Get a pointer to the version object that corresponds to the instance's version.
|
// Get a pointer to the version object that corresponds to the instance's version.
|
||||||
targetVersion = (MinecraftVersion *)MinecraftVersionList::getMainList().findVersion(m_inst->intendedVersionId());
|
targetVersion = (MinecraftVersion *)MinecraftVersionList::getMainList().findVersion(intendedVersion);
|
||||||
if(targetVersion == NULL)
|
if(targetVersion == nullptr)
|
||||||
{
|
{
|
||||||
|
// don't do anything if it was invalid
|
||||||
emit gameUpdateComplete();
|
emit gameUpdateComplete();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(m_inst->shouldUpdate())
|
||||||
|
{
|
||||||
|
versionFileStart();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
jarlibStart();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void OneSixUpdate::versionFileStart()
|
||||||
|
{
|
||||||
setStatus("Getting the version files from Mojang.");
|
setStatus("Getting the version files from Mojang.");
|
||||||
|
|
||||||
QString urlstr("http://s3.amazonaws.com/Minecraft.Download/versions/");
|
QString urlstr("http://s3.amazonaws.com/Minecraft.Download/versions/");
|
||||||
@ -59,71 +74,93 @@ void OneSixUpdate::executeTask()
|
|||||||
connect(specificVersionDownloadJob.data(), SIGNAL(failed()), SLOT(versionFileFailed()));
|
connect(specificVersionDownloadJob.data(), SIGNAL(failed()), SLOT(versionFileFailed()));
|
||||||
connect(specificVersionDownloadJob.data(), SIGNAL(progress(qint64,qint64)), SLOT(updateDownloadProgress(qint64,qint64)));
|
connect(specificVersionDownloadJob.data(), SIGNAL(progress(qint64,qint64)), SLOT(updateDownloadProgress(qint64,qint64)));
|
||||||
download_queue.enqueue(specificVersionDownloadJob);
|
download_queue.enqueue(specificVersionDownloadJob);
|
||||||
|
|
||||||
QEventLoop loop;
|
|
||||||
loop.exec();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void OneSixUpdate::versionFileFinished()
|
void OneSixUpdate::versionFileFinished()
|
||||||
{
|
{
|
||||||
JobPtr firstJob = specificVersionDownloadJob->getFirstJob();
|
JobPtr firstJob = specificVersionDownloadJob->getFirstJob();
|
||||||
auto DlJob = firstJob.dynamicCast<DownloadJob>();
|
auto DlJob = firstJob.dynamicCast<DownloadJob>();
|
||||||
FullVersionFactory parser;
|
|
||||||
auto version = parser.parse(DlJob->m_data);
|
|
||||||
|
|
||||||
if(!version)
|
|
||||||
{
|
|
||||||
error(parser.error_string);
|
|
||||||
exit(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
// save the version file in $instanceId/version.json and versions/$version/$version.json
|
|
||||||
QString version_id = targetVersion->descriptor();
|
QString version_id = targetVersion->descriptor();
|
||||||
QString inst_dir = m_inst->rootDir();
|
QString inst_dir = m_inst->rootDir();
|
||||||
QString version1 = PathCombine(inst_dir, "/version.json");
|
// save the version file in $instanceId/version.json
|
||||||
QString version2 = QString("versions/") + version_id + "/" + version_id + ".json";
|
{
|
||||||
DownloadJob::ensurePathExists(version1);
|
QString version1 = PathCombine(inst_dir, "/version.json");
|
||||||
DownloadJob::ensurePathExists(version2);
|
ensurePathExists(version1);
|
||||||
QFile vfile1 (version1);
|
QFile vfile1 (version1);
|
||||||
QFile vfile2 (version2);
|
vfile1.open(QIODevice::Truncate | QIODevice::WriteOnly );
|
||||||
vfile1.open(QIODevice::Truncate | QIODevice::WriteOnly );
|
vfile1.write(DlJob->m_data);
|
||||||
vfile2.open(QIODevice::Truncate | QIODevice::WriteOnly );
|
vfile1.close();
|
||||||
vfile1.write(DlJob->m_data);
|
}
|
||||||
vfile2.write(DlJob->m_data);
|
|
||||||
vfile1.close();
|
|
||||||
vfile2.close();
|
|
||||||
|
|
||||||
// download the right jar, save it in versions/$version/$version.jar
|
// save the version file in versions/$version/$version.json
|
||||||
QString urlstr("http://s3.amazonaws.com/Minecraft.Download/versions/");
|
/*
|
||||||
urlstr += targetVersion->descriptor() + "/" + targetVersion->descriptor() + ".jar";
|
//QString version2 = QString("versions/") + version_id + "/" + version_id + ".json";
|
||||||
QString targetstr ("versions/");
|
//ensurePathExists(version2);
|
||||||
targetstr += targetVersion->descriptor() + "/" + targetVersion->descriptor() + ".jar";
|
//QFile vfile2 (version2);
|
||||||
auto dljob = DownloadJob::create(QUrl(urlstr), targetstr);
|
//vfile2.open(QIODevice::Truncate | QIODevice::WriteOnly );
|
||||||
|
//vfile2.write(DlJob->m_data);
|
||||||
|
//vfile2.close();
|
||||||
|
*/
|
||||||
|
|
||||||
jarlibDownloadJob.reset(new JobList());
|
jarlibStart();
|
||||||
jarlibDownloadJob->add(dljob);
|
|
||||||
connect(jarlibDownloadJob.data(), SIGNAL(finished()), SLOT(jarlibFinished()));
|
|
||||||
connect(jarlibDownloadJob.data(), SIGNAL(failed()), SLOT(jarlibFailed()));
|
|
||||||
connect(jarlibDownloadJob.data(), SIGNAL(progress(qint64,qint64)), SLOT(updateDownloadProgress(qint64,qint64)));
|
|
||||||
// determine and download all the libraries, save them in libraries/whatever...
|
|
||||||
download_queue.enqueue(jarlibDownloadJob);
|
|
||||||
}
|
|
||||||
|
|
||||||
void OneSixUpdate::jarlibFinished()
|
|
||||||
{
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
void OneSixUpdate::jarlibFailed()
|
|
||||||
{
|
|
||||||
error("Failed to download the binary garbage. Try again. Maybe. IF YOU DARE");
|
|
||||||
exit(0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void OneSixUpdate::versionFileFailed()
|
void OneSixUpdate::versionFileFailed()
|
||||||
{
|
{
|
||||||
error("Failed to download the version description. Try again.");
|
error("Failed to download the version description. Try again.");
|
||||||
exit(0);
|
emitEnded();
|
||||||
|
}
|
||||||
|
|
||||||
|
void OneSixUpdate::jarlibStart()
|
||||||
|
{
|
||||||
|
OneSixInstance * inst = (OneSixInstance *) m_inst;
|
||||||
|
bool successful = inst->reloadFullVersion();
|
||||||
|
if(!successful)
|
||||||
|
{
|
||||||
|
error("Failed to load the version description file (version.json). It might be corrupted, missing or simply too new.");
|
||||||
|
emitEnded();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
QSharedPointer<FullVersion> version = inst->getFullVersion();
|
||||||
|
|
||||||
|
// download the right jar, save it in versions/$version/$version.jar
|
||||||
|
QString urlstr("http://s3.amazonaws.com/Minecraft.Download/versions/");
|
||||||
|
urlstr += version->id + "/" + version->id + ".jar";
|
||||||
|
QString targetstr ("versions/");
|
||||||
|
targetstr += version->id + "/" + version->id + ".jar";
|
||||||
|
|
||||||
|
auto dljob = DownloadJob::create(QUrl(urlstr), targetstr);
|
||||||
|
jarlibDownloadJob.reset(new JobList());
|
||||||
|
jarlibDownloadJob->add(dljob);
|
||||||
|
|
||||||
|
auto libs = version->getActiveNativeLibs();
|
||||||
|
libs.append(version->getActiveNormalLibs());
|
||||||
|
|
||||||
|
for(auto lib: libs)
|
||||||
|
{
|
||||||
|
QString download_path = lib->downloadPath();
|
||||||
|
QString storage_path = "libraries/" + lib->storagePath();
|
||||||
|
jarlibDownloadJob->add(DownloadJob::create(net_manager, download_path, storage_path));
|
||||||
|
}
|
||||||
|
connect(jarlibDownloadJob.data(), SIGNAL(finished()), SLOT(jarlibFinished()));
|
||||||
|
connect(jarlibDownloadJob.data(), SIGNAL(failed()), SLOT(jarlibFailed()));
|
||||||
|
connect(jarlibDownloadJob.data(), SIGNAL(progress(qint64,qint64)), SLOT(updateDownloadProgress(qint64,qint64)));
|
||||||
|
|
||||||
|
download_queue.enqueue(jarlibDownloadJob);
|
||||||
|
}
|
||||||
|
|
||||||
|
void OneSixUpdate::jarlibFinished()
|
||||||
|
{
|
||||||
|
emit gameUpdateComplete();
|
||||||
|
emitEnded();
|
||||||
|
}
|
||||||
|
|
||||||
|
void OneSixUpdate::jarlibFailed()
|
||||||
|
{
|
||||||
|
error("Failed to download the binary garbage. Try again. Maybe. IF YOU DARE");
|
||||||
|
emitEnded();
|
||||||
}
|
}
|
||||||
|
|
||||||
void OneSixUpdate::error(const QString &msg)
|
void OneSixUpdate::error(const QString &msg)
|
||||||
|
@ -46,16 +46,18 @@ public slots:
|
|||||||
private slots:
|
private slots:
|
||||||
void updateDownloadProgress(qint64 current, qint64 total);
|
void updateDownloadProgress(qint64 current, qint64 total);
|
||||||
|
|
||||||
|
void versionFileStart();
|
||||||
void versionFileFinished();
|
void versionFileFinished();
|
||||||
void versionFileFailed();
|
void versionFileFailed();
|
||||||
|
|
||||||
|
void jarlibStart();
|
||||||
void jarlibFinished();
|
void jarlibFinished();
|
||||||
void jarlibFailed();
|
void jarlibFailed();
|
||||||
|
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
/*!
|
/*!
|
||||||
* \brief Signal emitted when the game update is complete.
|
* \brief Signal emitted when the game update is complete.
|
||||||
* \param response The login response received from login task.
|
|
||||||
*/
|
*/
|
||||||
void gameUpdateComplete();
|
void gameUpdateComplete();
|
||||||
|
|
||||||
@ -70,6 +72,7 @@ private:
|
|||||||
|
|
||||||
QString m_subStatusMsg;
|
QString m_subStatusMsg;
|
||||||
|
|
||||||
|
QSharedPointer<QNetworkAccessManager> net_manager {new QNetworkAccessManager()};
|
||||||
JobListPtr legacyDownloadJob;
|
JobListPtr legacyDownloadJob;
|
||||||
JobListPtr specificVersionDownloadJob;
|
JobListPtr specificVersionDownloadJob;
|
||||||
JobListPtr jarlibDownloadJob;
|
JobListPtr jarlibDownloadJob;
|
||||||
|
@ -64,12 +64,51 @@ void Library::finalize()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Library::setName ( QString name )
|
||||||
|
{
|
||||||
|
m_name = name;
|
||||||
|
}
|
||||||
|
void Library::setBaseUrl ( QString base_url )
|
||||||
|
{
|
||||||
|
m_base_url = base_url;
|
||||||
|
}
|
||||||
|
void Library::setIsNative()
|
||||||
|
{
|
||||||
|
m_is_native = true;
|
||||||
|
}
|
||||||
|
void Library::addNative ( OpSys os, QString suffix )
|
||||||
|
{
|
||||||
|
m_is_native = true;
|
||||||
|
m_native_suffixes[os] = suffix;
|
||||||
|
}
|
||||||
|
void Library::setRules ( QList< QSharedPointer< Rule > > rules )
|
||||||
|
{
|
||||||
|
m_rules = rules;
|
||||||
|
}
|
||||||
|
bool Library::isActive()
|
||||||
|
{
|
||||||
|
return m_is_active;
|
||||||
|
}
|
||||||
|
bool Library::isNative()
|
||||||
|
{
|
||||||
|
return m_is_native;
|
||||||
|
}
|
||||||
|
QString Library::downloadPath()
|
||||||
|
{
|
||||||
|
return m_download_path;
|
||||||
|
}
|
||||||
|
QString Library::storagePath()
|
||||||
|
{
|
||||||
|
return m_storage_path;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
QList<QSharedPointer<Library> > FullVersion::getActiveNormalLibs()
|
QList<QSharedPointer<Library> > FullVersion::getActiveNormalLibs()
|
||||||
{
|
{
|
||||||
QList<QSharedPointer<Library> > output;
|
QList<QSharedPointer<Library> > output;
|
||||||
for ( auto lib: libraries )
|
for ( auto lib: libraries )
|
||||||
{
|
{
|
||||||
if (lib->getIsActive() && !lib->getIsNative())
|
if (lib->isActive() && !lib->isNative())
|
||||||
{
|
{
|
||||||
output.append(lib);
|
output.append(lib);
|
||||||
}
|
}
|
||||||
@ -82,10 +121,12 @@ QList<QSharedPointer<Library> > FullVersion::getActiveNativeLibs()
|
|||||||
QList<QSharedPointer<Library> > output;
|
QList<QSharedPointer<Library> > output;
|
||||||
for ( auto lib: libraries )
|
for ( auto lib: libraries )
|
||||||
{
|
{
|
||||||
if (lib->getIsActive() && lib->getIsNative())
|
if (lib->isActive() && lib->isNative())
|
||||||
{
|
{
|
||||||
output.append(lib);
|
output.append(lib);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return output;
|
return output;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -107,11 +107,11 @@ private:
|
|||||||
QString m_storage_path;
|
QString m_storage_path;
|
||||||
/// where to download the lib from
|
/// where to download the lib from
|
||||||
QString m_download_path;
|
QString m_download_path;
|
||||||
/// is this lib actuall active on the current OS?
|
/// is this lib actually active on the current OS?
|
||||||
bool m_is_active;
|
bool m_is_active;
|
||||||
|
/// is the library a native?
|
||||||
// native lib?
|
|
||||||
bool m_is_native;
|
bool m_is_native;
|
||||||
|
/// native suffixes per OS
|
||||||
QMap<OpSys, QString> m_native_suffixes;
|
QMap<OpSys, QString> m_native_suffixes;
|
||||||
public:
|
public:
|
||||||
QStringList extract_excludes;
|
QStringList extract_excludes;
|
||||||
@ -133,62 +133,25 @@ public:
|
|||||||
*/
|
*/
|
||||||
void finalize();
|
void finalize();
|
||||||
|
|
||||||
|
/// Set the library composite name
|
||||||
/**
|
void setName(QString name);
|
||||||
* Set the library composite name
|
/// Set the url base for downloads
|
||||||
*/
|
void setBaseUrl(QString base_url);
|
||||||
void setName(QString name)
|
/// Call this to mark the library as 'native' (it's a zip archive with DLLs)
|
||||||
{
|
void setIsNative();
|
||||||
m_name = name;
|
/// Attach a name suffix to the specified OS native
|
||||||
}
|
void addNative(OpSys os, QString suffix);
|
||||||
|
/// Set the load rules
|
||||||
/**
|
void setRules(QList<QSharedPointer<Rule> > rules);
|
||||||
* Set the url base for downloads
|
|
||||||
*/
|
|
||||||
void setBaseUrl(QString base_url)
|
|
||||||
{
|
|
||||||
m_base_url = base_url;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Call this to mark the library as 'native' (it's a zip archive with DLLs)
|
|
||||||
*/
|
|
||||||
void setIsNative()
|
|
||||||
{
|
|
||||||
m_is_native = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Attach a name suffix to the specified OS native
|
|
||||||
*/
|
|
||||||
void addNative(OpSys os, QString suffix)
|
|
||||||
{
|
|
||||||
m_is_native = true;
|
|
||||||
m_native_suffixes[os] = suffix;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the load rules
|
|
||||||
*/
|
|
||||||
void setRules(QList<QSharedPointer<Rule> > rules)
|
|
||||||
{
|
|
||||||
m_rules = rules;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/// Returns true if the library should be loaded (or extracted, in case of natives)
|
||||||
* Returns true if the library should be loaded (or extracted, in case of natives)
|
bool isActive();
|
||||||
*/
|
/// Returns true if the library is native
|
||||||
bool getIsActive()
|
bool isNative();
|
||||||
{
|
/// Get the URL to download the library from
|
||||||
return m_is_active;
|
QString downloadPath();
|
||||||
}
|
/// Get the relative path where the library should be saved
|
||||||
/**
|
QString storagePath();
|
||||||
* Returns true if the library is native
|
|
||||||
*/
|
|
||||||
bool getIsNative()
|
|
||||||
{
|
|
||||||
return m_is_native;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -172,24 +172,21 @@ MCVListLoadTask::~MCVListLoadTask()
|
|||||||
|
|
||||||
void MCVListLoadTask::executeTask()
|
void MCVListLoadTask::executeTask()
|
||||||
{
|
{
|
||||||
// NOTE: this executes in the QThread
|
|
||||||
setStatus("Loading instance version list...");
|
setStatus("Loading instance version list...");
|
||||||
netMgr = new QNetworkAccessManager();
|
netMgr = new QNetworkAccessManager();
|
||||||
vlistReply = netMgr->get(QNetworkRequest(QUrl(QString(MCVLIST_URLBASE) + "versions.json")));
|
vlistReply = netMgr->get(QNetworkRequest(QUrl(QString(MCVLIST_URLBASE) + "versions.json")));
|
||||||
connect(vlistReply, SIGNAL(finished()), this, SLOT(list_downloaded()));
|
connect(vlistReply, SIGNAL(finished()), this, SLOT(list_downloaded()));
|
||||||
exec();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void MCVListLoadTask::list_downloaded()
|
void MCVListLoadTask::list_downloaded()
|
||||||
{
|
{
|
||||||
// NOTE: this executes in the main thread
|
|
||||||
|
|
||||||
if(vlistReply->error() != QNetworkReply::QNetworkReply::NoError)
|
if(vlistReply->error() != QNetworkReply::QNetworkReply::NoError)
|
||||||
{
|
{
|
||||||
qDebug() << "Failed to load Minecraft main version list" << vlistReply->errorString();
|
qDebug() << "Failed to load Minecraft main version list" << vlistReply->errorString();
|
||||||
vlistReply->deleteLater();
|
vlistReply->deleteLater();
|
||||||
exit(0);
|
emitEnded();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
QJsonParseError jsonError;
|
QJsonParseError jsonError;
|
||||||
@ -199,13 +196,15 @@ void MCVListLoadTask::list_downloaded()
|
|||||||
if (jsonError.error != QJsonParseError::NoError)
|
if (jsonError.error != QJsonParseError::NoError)
|
||||||
{
|
{
|
||||||
qDebug() << "Error parsing version list JSON:" << jsonError.errorString();
|
qDebug() << "Error parsing version list JSON:" << jsonError.errorString();
|
||||||
exit(0);
|
emitEnded();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!jsonDoc.isObject())
|
if(!jsonDoc.isObject())
|
||||||
{
|
{
|
||||||
qDebug() << "Error parsing version list JSON: " << "jsonDoc is not an object";
|
qDebug() << "Error parsing version list JSON: " << "jsonDoc is not an object";
|
||||||
exit(0);
|
emitEnded();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
QJsonObject root = jsonDoc.object();
|
QJsonObject root = jsonDoc.object();
|
||||||
@ -214,7 +213,8 @@ void MCVListLoadTask::list_downloaded()
|
|||||||
if(!root.value("latest").isObject())
|
if(!root.value("latest").isObject())
|
||||||
{
|
{
|
||||||
qDebug() << "Error parsing version list JSON: " << "version list is missing 'latest' object";
|
qDebug() << "Error parsing version list JSON: " << "version list is missing 'latest' object";
|
||||||
exit(0);
|
emitEnded();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
QJsonObject latest = root.value("latest").toObject();
|
QJsonObject latest = root.value("latest").toObject();
|
||||||
@ -224,19 +224,22 @@ void MCVListLoadTask::list_downloaded()
|
|||||||
if(latestReleaseID.isEmpty())
|
if(latestReleaseID.isEmpty())
|
||||||
{
|
{
|
||||||
qDebug() << "Error parsing version list JSON: " << "latest release field is missing";
|
qDebug() << "Error parsing version list JSON: " << "latest release field is missing";
|
||||||
exit(0);
|
emitEnded();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
if(latestSnapshotID.isEmpty())
|
if(latestSnapshotID.isEmpty())
|
||||||
{
|
{
|
||||||
qDebug() << "Error parsing version list JSON: " << "latest snapshot field is missing";
|
qDebug() << "Error parsing version list JSON: " << "latest snapshot field is missing";
|
||||||
exit(0);
|
emitEnded();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now, get the array of versions.
|
// Now, get the array of versions.
|
||||||
if(!root.value("versions").isArray())
|
if(!root.value("versions").isArray())
|
||||||
{
|
{
|
||||||
qDebug() << "Error parsing version list JSON: " << "version list object is missing 'versions' array";
|
qDebug() << "Error parsing version list JSON: " << "version list object is missing 'versions' array";
|
||||||
exit(0);
|
emitEnded();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
QJsonArray versions = root.value("versions").toArray();
|
QJsonArray versions = root.value("versions").toArray();
|
||||||
|
|
||||||
@ -303,7 +306,8 @@ void MCVListLoadTask::list_downloaded()
|
|||||||
#ifdef PRINT_VERSIONS
|
#ifdef PRINT_VERSIONS
|
||||||
m_list->printToStdOut();
|
m_list->printToStdOut();
|
||||||
#endif
|
#endif
|
||||||
exit(1);
|
emitEnded();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: we should have a local cache of the version list and a local cache of version data
|
// FIXME: we should have a local cache of the version list and a local cache of version data
|
||||||
|
@ -27,16 +27,14 @@
|
|||||||
LoginTask::LoginTask( const UserInfo& uInfo, QObject* parent ) :
|
LoginTask::LoginTask( const UserInfo& uInfo, QObject* parent ) :
|
||||||
Task(parent), uInfo(uInfo)
|
Task(parent), uInfo(uInfo)
|
||||||
{
|
{
|
||||||
|
netMgr.reset(new QNetworkAccessManager());
|
||||||
}
|
}
|
||||||
|
|
||||||
void LoginTask::executeTask()
|
void LoginTask::executeTask()
|
||||||
{
|
{
|
||||||
setStatus("Logging in...");
|
setStatus("Logging in...");
|
||||||
|
|
||||||
QNetworkAccessManager netMgr;
|
connect(netMgr.data(), SIGNAL(finished(QNetworkReply*)), this, SLOT(processNetReply(QNetworkReply*)));
|
||||||
connect(&netMgr, SIGNAL(finished(QNetworkReply*)),
|
|
||||||
SLOT(processNetReply(QNetworkReply*)));
|
|
||||||
|
|
||||||
QUrl loginURL("https://login.minecraft.net/");
|
QUrl loginURL("https://login.minecraft.net/");
|
||||||
QNetworkRequest netRequest(loginURL);
|
QNetworkRequest netRequest(loginURL);
|
||||||
@ -47,8 +45,7 @@ void LoginTask::executeTask()
|
|||||||
params.addQueryItem("password", uInfo.password);
|
params.addQueryItem("password", uInfo.password);
|
||||||
params.addQueryItem("version", "13");
|
params.addQueryItem("version", "13");
|
||||||
|
|
||||||
netReply = netMgr.post(netRequest, params.query(QUrl::EncodeSpaces).toUtf8());
|
netReply = netMgr->post(netRequest, params.query(QUrl::EncodeSpaces).toUtf8());
|
||||||
exec();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void LoginTask::processNetReply(QNetworkReply *reply)
|
void LoginTask::processNetReply(QNetworkReply *reply)
|
||||||
@ -115,6 +112,5 @@ void LoginTask::processNetReply(QNetworkReply *reply)
|
|||||||
emit loginFailed("Login failed: " + reply->errorString());
|
emit loginFailed("Login failed: " + reply->errorString());
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
emitEnded();
|
||||||
quit();
|
|
||||||
}
|
}
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
#define LOGINTASK_H
|
#define LOGINTASK_H
|
||||||
|
|
||||||
#include "Task.h"
|
#include "Task.h"
|
||||||
|
#include <QSharedPointer>
|
||||||
#include "libmmc_config.h"
|
#include "libmmc_config.h"
|
||||||
|
|
||||||
struct UserInfo
|
struct UserInfo
|
||||||
@ -33,7 +33,7 @@ struct LoginResponse
|
|||||||
qint64 latestVersion;
|
qint64 latestVersion;
|
||||||
};
|
};
|
||||||
|
|
||||||
//class QNetworkAccessManager;
|
class QNetworkAccessManager;
|
||||||
class QNetworkReply;
|
class QNetworkReply;
|
||||||
|
|
||||||
class LIBMULTIMC_EXPORT LoginTask : public Task
|
class LIBMULTIMC_EXPORT LoginTask : public Task
|
||||||
@ -54,6 +54,8 @@ protected:
|
|||||||
|
|
||||||
QNetworkReply* netReply;
|
QNetworkReply* netReply;
|
||||||
UserInfo uInfo;
|
UserInfo uInfo;
|
||||||
|
private:
|
||||||
|
QSharedPointer<QNetworkAccessManager> netMgr;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // LOGINTASK_H
|
#endif // LOGINTASK_H
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
#include "Task.h"
|
#include "Task.h"
|
||||||
|
|
||||||
Task::Task(QObject *parent) :
|
Task::Task(QObject *parent) :
|
||||||
QThread(parent)
|
QObject(parent)
|
||||||
{
|
{
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -49,29 +49,31 @@ void Task::setProgress(int progress)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void Task::startTask()
|
void Task::startTask()
|
||||||
{
|
|
||||||
start();
|
|
||||||
}
|
|
||||||
|
|
||||||
void Task::run()
|
|
||||||
{
|
{
|
||||||
emitStarted();
|
emitStarted();
|
||||||
executeTask();
|
executeTask();
|
||||||
emitEnded();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Task::emitStarted()
|
void Task::emitStarted()
|
||||||
{
|
{
|
||||||
|
running = true;
|
||||||
emit started();
|
emit started();
|
||||||
emit started(this);
|
emit started(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Task::emitEnded()
|
void Task::emitEnded()
|
||||||
{
|
{
|
||||||
|
running = false;
|
||||||
emit ended();
|
emit ended();
|
||||||
emit ended(this);
|
emit ended(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool Task::isRunning() const
|
||||||
|
{
|
||||||
|
return running;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void Task::emitStatusChange(const QString &status)
|
void Task::emitStatusChange(const QString &status)
|
||||||
{
|
{
|
||||||
emit statusChanged(status);
|
emit statusChanged(status);
|
||||||
|
@ -17,12 +17,11 @@
|
|||||||
#define TASK_H
|
#define TASK_H
|
||||||
|
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QThread>
|
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
|
||||||
#include "libmmc_config.h"
|
#include "libmmc_config.h"
|
||||||
|
|
||||||
class LIBMULTIMC_EXPORT Task : public QThread
|
class LIBMULTIMC_EXPORT Task : public QObject
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
@ -34,6 +33,8 @@ public:
|
|||||||
QString getStatus() const;
|
QString getStatus() const;
|
||||||
int getProgress() const;
|
int getProgress() const;
|
||||||
|
|
||||||
|
bool isRunning() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \brief Calculates and sets the task's progress based on the number of parts completed out of the total number to complete.
|
* \brief Calculates and sets the task's progress based on the number of parts completed out of the total number to complete.
|
||||||
* This is essentially just shorthand for setProgress((parts / whole) * 100);
|
* This is essentially just shorthand for setProgress((parts / whole) * 100);
|
||||||
@ -43,7 +44,7 @@ public:
|
|||||||
*/
|
*/
|
||||||
void calcProgress(int parts, int whole);
|
void calcProgress(int parts, int whole);
|
||||||
|
|
||||||
public slots:
|
protected slots:
|
||||||
void setStatus(const QString& status);
|
void setStatus(const QString& status);
|
||||||
void setProgress(int progress);
|
void setProgress(int progress);
|
||||||
|
|
||||||
@ -54,7 +55,6 @@ signals:
|
|||||||
void started();
|
void started();
|
||||||
void ended();
|
void ended();
|
||||||
|
|
||||||
|
|
||||||
void statusChanged(Task* task, const QString& status);
|
void statusChanged(Task* task, const QString& status);
|
||||||
void progressChanged(Task* task, int progress);
|
void progressChanged(Task* task, int progress);
|
||||||
|
|
||||||
@ -62,7 +62,6 @@ signals:
|
|||||||
void progressChanged(int progress);
|
void progressChanged(int progress);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
virtual void run();
|
|
||||||
virtual void executeTask() = 0;
|
virtual void executeTask() = 0;
|
||||||
|
|
||||||
virtual void emitStarted();
|
virtual void emitStarted();
|
||||||
@ -73,6 +72,7 @@ protected:
|
|||||||
|
|
||||||
QString status;
|
QString status;
|
||||||
int progress;
|
int progress;
|
||||||
|
bool running = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // TASK_H
|
#endif // TASK_H
|
||||||
|
@ -26,8 +26,6 @@ public:
|
|||||||
QString expected_md5 = QString()
|
QString expected_md5 = QString()
|
||||||
);
|
);
|
||||||
|
|
||||||
public:
|
|
||||||
static bool ensurePathExists(QString filenamepath);
|
|
||||||
public slots:
|
public slots:
|
||||||
virtual void start();
|
virtual void start();
|
||||||
|
|
||||||
|
@ -29,4 +29,6 @@ LIBUTIL_EXPORT QString RemoveInvalidFilenameChars(QString string, QChar replaceW
|
|||||||
|
|
||||||
LIBUTIL_EXPORT QString DirNameFromString(QString string, QString inDir = ".");
|
LIBUTIL_EXPORT QString DirNameFromString(QString string, QString inDir = ".");
|
||||||
|
|
||||||
|
LIBUTIL_EXPORT bool ensurePathExists(QString filenamepath);
|
||||||
|
|
||||||
#endif // PATHUTILS_H
|
#endif // PATHUTILS_H
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
#include "include/dlqueue.h"
|
#include "include/dlqueue.h"
|
||||||
|
#include <include/pathutils.h>
|
||||||
|
|
||||||
DownloadJob::DownloadJob (QUrl url,
|
DownloadJob::DownloadJob (QUrl url,
|
||||||
QString target_path,
|
QString target_path,
|
||||||
@ -48,13 +49,6 @@ JobPtr DownloadJob::create (QSharedPointer<QNetworkAccessManager> net_mgr,
|
|||||||
return JobPtr ( new DownloadJob ( net_mgr, url, target_path, expected_md5 ) );
|
return JobPtr ( new DownloadJob ( net_mgr, url, target_path, expected_md5 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DownloadJob::ensurePathExists(QString filenamepath)
|
|
||||||
{
|
|
||||||
QFileInfo a ( filenamepath );
|
|
||||||
QDir dir;
|
|
||||||
return (dir.mkpath ( a.path() ));
|
|
||||||
}
|
|
||||||
|
|
||||||
void DownloadJob::start()
|
void DownloadJob::start()
|
||||||
{
|
{
|
||||||
if ( m_save_to_file )
|
if ( m_save_to_file )
|
||||||
|
@ -65,3 +65,11 @@ QString DirNameFromString(QString string, QString inDir)
|
|||||||
}
|
}
|
||||||
return dirName;
|
return dirName;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool ensurePathExists(QString filenamepath)
|
||||||
|
{
|
||||||
|
QFileInfo a ( filenamepath );
|
||||||
|
QDir dir;
|
||||||
|
return (dir.mkpath ( a.path() ));
|
||||||
|
}
|
||||||
|
|
||||||
|
@ -384,6 +384,53 @@ QStringList JlCompress::extractFiles(QString fileCompressed, QStringList files,
|
|||||||
return extracted;
|
return extracted;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QStringList JlCompress::extractWithExceptions(QString fileCompressed, QString dir, QStringList exceptions)
|
||||||
|
{
|
||||||
|
QuaZip zip(fileCompressed);
|
||||||
|
if(!zip.open(QuaZip::mdUnzip))
|
||||||
|
{
|
||||||
|
return QStringList();
|
||||||
|
}
|
||||||
|
|
||||||
|
QDir directory(dir);
|
||||||
|
QStringList extracted;
|
||||||
|
if (!zip.goToFirstFile())
|
||||||
|
{
|
||||||
|
return QStringList();
|
||||||
|
}
|
||||||
|
do
|
||||||
|
{
|
||||||
|
QString name = zip.getCurrentFileName();
|
||||||
|
bool ok = true;
|
||||||
|
for(auto str: exceptions)
|
||||||
|
{
|
||||||
|
if(name.startsWith(str))
|
||||||
|
{
|
||||||
|
ok = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(!ok)
|
||||||
|
continue;
|
||||||
|
QString absFilePath = directory.absoluteFilePath(name);
|
||||||
|
if (!JlCompress::extractFile(&zip, "", absFilePath))
|
||||||
|
{
|
||||||
|
JlCompress::removeFile(extracted);
|
||||||
|
return QStringList();
|
||||||
|
}
|
||||||
|
extracted.append(absFilePath);
|
||||||
|
} while (zip.goToNextFile());
|
||||||
|
|
||||||
|
zip.close();
|
||||||
|
if(zip.getZipError()!=0)
|
||||||
|
{
|
||||||
|
JlCompress::removeFile(extracted);
|
||||||
|
return QStringList();
|
||||||
|
}
|
||||||
|
|
||||||
|
return extracted;
|
||||||
|
}
|
||||||
|
|
||||||
/**OK
|
/**OK
|
||||||
* Estrae il file fileCompressed nella cartella dir.
|
* Estrae il file fileCompressed nella cartella dir.
|
||||||
* Se dir = "" allora il file viene estratto nella cartella corrente.
|
* Se dir = "" allora il file viene estratto nella cartella corrente.
|
||||||
|
@ -102,6 +102,15 @@ public:
|
|||||||
\return The list of the full paths of the files extracted, empty on failure.
|
\return The list of the full paths of the files extracted, empty on failure.
|
||||||
*/
|
*/
|
||||||
static QStringList extractDir(QString fileCompressed, QString dir = QString());
|
static QStringList extractDir(QString fileCompressed, QString dir = QString());
|
||||||
|
/// Extract a whole archive, with a list of exceptions (prefixes to ignore).
|
||||||
|
/**
|
||||||
|
\param fileCompressed The name of the archive.
|
||||||
|
\param dir The directory to extract to, the current directory if
|
||||||
|
left empty.
|
||||||
|
\param exceptions The list of exception prefixes
|
||||||
|
\return The list of the full paths of the files extracted, empty on failure.
|
||||||
|
*/
|
||||||
|
static QStringList extractWithExceptions(QString fileCompressed, QString dir, QStringList exceptions);
|
||||||
/// Get the file list.
|
/// Get the file list.
|
||||||
/**
|
/**
|
||||||
\return The list of the files in the archive, or, more precisely, the
|
\return The list of the files in the archive, or, more precisely, the
|
||||||
|
Loading…
x
Reference in New Issue
Block a user