2013-02-21 18:10:17 -06:00
/* Copyright 2013 MultiMC Contributors
*
* 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
2013-09-22 04:21:36 +02:00
*
2013-02-21 18:10:17 -06:00
* 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 .
*/
2013-11-04 02:53:05 +01:00
2013-09-08 02:15:20 +02:00
# include "MultiMC.h"
2013-08-04 14:46:33 +02:00
# include "OneSixUpdate.h"
2013-02-21 18:10:17 -06:00
2013-05-08 12:56:43 -05:00
# include <QtNetwork>
# include <QFile>
# include <QFileInfo>
# include <QTextStream>
# include <QDataStream>
2013-08-03 23:58:39 +02:00
# include "BaseInstance.h"
2013-07-29 00:59:35 +02:00
# include "lists/MinecraftVersionList.h"
2014-03-01 23:06:47 +01:00
# include "VersionFinal.h"
2013-09-11 23:43:17 +02:00
# include "OneSixLibrary.h"
2013-08-05 03:29:50 +02:00
# include "OneSixInstance.h"
2013-11-17 11:44:18 +01:00
# include "net/ForgeMirrors.h"
2013-12-13 14:58:11 +00:00
# include "net/URLConstants.h"
2013-12-10 07:12:52 +01:00
# include "assets/AssetsUtils.h"
2013-05-08 12:56:43 -05:00
# include "pathutils.h"
2013-11-25 00:46:52 +01:00
# include <JlCompress.h>
2013-05-08 12:56:43 -05:00
2014-02-21 19:15:59 +01:00
OneSixUpdate : : OneSixUpdate ( OneSixInstance * inst , QObject * parent )
2014-01-27 03:00:49 +01:00
: Task ( parent ) , m_inst ( inst )
2013-09-22 04:21:36 +02:00
{
}
2013-05-08 12:56:43 -05:00
2013-08-04 14:46:33 +02:00
void OneSixUpdate : : executeTask ( )
2013-05-08 12:56:43 -05:00
{
2013-08-05 03:29:50 +02:00
QString intendedVersion = m_inst - > intendedVersionId ( ) ;
2013-09-22 04:21:36 +02:00
2013-08-12 00:39:19 +02:00
// Make directories
QDir mcDir ( m_inst - > minecraftRoot ( ) ) ;
if ( ! mcDir . exists ( ) & & ! mcDir . mkpath ( " . " ) )
{
2014-03-09 23:42:25 +01:00
emitFailed ( tr ( " Failed to create folder for minecraft binaries. " ) ) ;
2013-08-12 00:39:19 +02:00
return ;
}
2013-09-22 04:21:36 +02:00
if ( m_inst - > shouldUpdate ( ) )
2013-08-05 03:29:50 +02:00
{
2013-11-25 00:46:52 +01:00
// Get a pointer to the version object that corresponds to the instance's version.
targetVersion = std : : dynamic_pointer_cast < MinecraftVersion > (
MMC - > minecraftlist ( ) - > findVersion ( intendedVersion ) ) ;
if ( targetVersion = = nullptr )
{
// don't do anything if it was invalid
2014-03-09 23:42:25 +01:00
emitFailed ( tr ( " The specified Minecraft version is invalid. Choose a different one. " ) ) ;
2013-11-25 00:46:52 +01:00
return ;
}
2013-08-05 03:29:50 +02:00
versionFileStart ( ) ;
}
else
{
jarlibStart ( ) ;
}
}
void OneSixUpdate : : versionFileStart ( )
{
2014-02-21 19:15:59 +01:00
if ( m_inst - > providesVersionFile ( ) )
{
jarlibStart ( ) ;
return ;
}
2013-11-25 00:46:52 +01:00
QLOG_INFO ( ) < < m_inst - > name ( ) < < " : getting version file. " ;
2013-12-23 15:46:01 +00:00
setStatus ( tr ( " Getting the version files from Mojang... " ) ) ;
2013-09-22 04:21:36 +02:00
2014-01-14 01:13:35 +01:00
QString urlstr = " http:// " + URLConstants : : AWS_DOWNLOAD_VERSIONS +
targetVersion - > descriptor ( ) + " / " + targetVersion - > descriptor ( ) + " .json " ;
2013-10-26 19:55:48 +02:00
auto job = new NetJob ( " Version index " ) ;
job - > addNetAction ( ByteArrayDownload : : make ( QUrl ( urlstr ) ) ) ;
2013-09-08 02:15:20 +02:00
specificVersionDownloadJob . reset ( job ) ;
2013-10-06 01:13:40 +02:00
connect ( specificVersionDownloadJob . get ( ) , SIGNAL ( succeeded ( ) ) , SLOT ( versionFileFinished ( ) ) ) ;
connect ( specificVersionDownloadJob . get ( ) , SIGNAL ( failed ( ) ) , SLOT ( versionFileFailed ( ) ) ) ;
connect ( specificVersionDownloadJob . get ( ) , SIGNAL ( progress ( qint64 , qint64 ) ) ,
2013-09-22 04:21:36 +02:00
SIGNAL ( progress ( qint64 , qint64 ) ) ) ;
2013-09-02 00:25:40 +02:00
specificVersionDownloadJob - > start ( ) ;
2013-07-07 23:51:26 +02:00
}
2013-08-04 14:46:33 +02:00
void OneSixUpdate : : versionFileFinished ( )
2013-07-07 23:51:26 +02:00
{
2013-10-26 19:55:48 +02:00
NetActionPtr DlJob = specificVersionDownloadJob - > first ( ) ;
2013-09-22 04:21:36 +02:00
2013-09-16 00:54:39 +02:00
QString version_id = targetVersion - > descriptor ( ) ;
2013-08-12 00:39:19 +02:00
QString inst_dir = m_inst - > instanceRoot ( ) ;
2013-08-05 03:29:50 +02:00
// save the version file in $instanceId/version.json
2013-07-07 23:51:26 +02:00
{
2013-09-22 04:21:36 +02:00
QString version1 = PathCombine ( inst_dir , " /version.json " ) ;
2013-08-24 03:09:46 +02:00
ensureFilePathExists ( version1 ) ;
2013-08-14 08:13:41 +02:00
// FIXME: detect errors here, download to a temp file, swap
2013-09-22 04:21:36 +02:00
QSaveFile vfile1 ( version1 ) ;
if ( ! vfile1 . open ( QIODevice : : Truncate | QIODevice : : WriteOnly ) )
2013-09-16 00:54:39 +02:00
{
2014-03-09 23:42:25 +01:00
emitFailed ( tr ( " Can't open %1 for writing. " ) . arg ( version1 ) ) ;
2013-09-16 00:54:39 +02:00
return ;
}
2013-10-06 01:13:40 +02:00
auto data = std : : dynamic_pointer_cast < ByteArrayDownload > ( DlJob ) - > m_data ;
2013-09-16 00:54:39 +02:00
qint64 actual = 0 ;
2013-09-22 04:21:36 +02:00
if ( ( actual = vfile1 . write ( data ) ) ! = data . size ( ) )
2013-09-16 00:54:39 +02:00
{
2014-03-09 23:42:25 +01:00
emitFailed ( tr ( " Failed to write into %1. Written %2 out of %3. " ) . arg ( version1 ) . arg ( actual ) . arg ( data . size ( ) ) ) ;
2013-09-16 00:54:39 +02:00
return ;
}
2013-09-22 04:21:36 +02:00
if ( ! vfile1 . commit ( ) )
2013-09-16 00:54:39 +02:00
{
2014-03-09 23:42:25 +01:00
emitFailed ( tr ( " Can't commit changes to %1 " ) . arg ( version1 ) ) ;
2013-09-16 00:54:39 +02:00
return ;
}
2013-07-07 23:51:26 +02:00
}
2013-09-22 04:21:36 +02:00
2013-08-14 08:13:41 +02:00
// the version is downloaded safely. update is 'done' at this point
m_inst - > setShouldUpdate ( false ) ;
2013-09-22 04:21:36 +02:00
2013-09-16 00:54:39 +02:00
// delete any custom version inside the instance (it's no longer relevant, we did an update)
2013-09-22 04:21:36 +02:00
QString custom = PathCombine ( inst_dir , " /custom.json " ) ;
2013-09-16 00:54:39 +02:00
QFile finfo ( custom ) ;
2013-09-22 04:21:36 +02:00
if ( finfo . exists ( ) )
2013-09-16 00:54:39 +02:00
{
finfo . remove ( ) ;
}
2014-03-09 23:42:25 +01:00
// NOTE: Version is reloaded in jarlibStart
2014-01-14 01:13:35 +01:00
jarlibStart ( ) ;
2013-08-05 03:29:50 +02:00
}
void OneSixUpdate : : versionFileFailed ( )
{
2014-03-09 23:42:25 +01:00
emitFailed ( tr ( " Failed to download the version description. Try again. " ) ) ;
2013-08-05 03:29:50 +02:00
}
2013-12-10 07:12:52 +01:00
void OneSixUpdate : : assetIndexStart ( )
{
2013-12-23 15:46:01 +00:00
setStatus ( tr ( " Updating assets index... " ) ) ;
2013-12-10 07:12:52 +01:00
OneSixInstance * inst = ( OneSixInstance * ) m_inst ;
2014-03-01 23:06:47 +01:00
std : : shared_ptr < VersionFinal > version = inst - > getFullVersion ( ) ;
2013-12-10 07:12:52 +01:00
QString assetName = version - > assets ;
2013-12-13 14:58:11 +00:00
QUrl indexUrl = " http:// " + URLConstants : : AWS_DOWNLOAD_INDEXES + assetName + " .json " ;
2013-12-10 07:12:52 +01:00
QString localPath = assetName + " .json " ;
2014-03-23 19:07:13 +01:00
auto job = new NetJob ( tr ( " Asset index for %1 " ) . arg ( inst - > name ( ) ) ) ;
2013-12-10 07:12:52 +01:00
auto metacache = MMC - > metacache ( ) ;
auto entry = metacache - > resolveEntry ( " asset_indexes " , localPath ) ;
job - > addNetAction ( CacheDownload : : make ( indexUrl , entry ) ) ;
jarlibDownloadJob . reset ( job ) ;
connect ( jarlibDownloadJob . get ( ) , SIGNAL ( succeeded ( ) ) , SLOT ( assetIndexFinished ( ) ) ) ;
connect ( jarlibDownloadJob . get ( ) , SIGNAL ( failed ( ) ) , SLOT ( assetIndexFailed ( ) ) ) ;
connect ( jarlibDownloadJob . get ( ) , SIGNAL ( progress ( qint64 , qint64 ) ) ,
SIGNAL ( progress ( qint64 , qint64 ) ) ) ;
jarlibDownloadJob - > start ( ) ;
}
void OneSixUpdate : : assetIndexFinished ( )
{
AssetsIndex index ;
OneSixInstance * inst = ( OneSixInstance * ) m_inst ;
2014-03-01 23:06:47 +01:00
std : : shared_ptr < VersionFinal > version = inst - > getFullVersion ( ) ;
2013-12-10 07:12:52 +01:00
QString assetName = version - > assets ;
QString asset_fname = " assets/indexes/ " + assetName + " .json " ;
if ( ! AssetsUtils : : loadAssetsIndexJson ( asset_fname , & index ) )
{
2014-03-09 23:42:25 +01:00
emitFailed ( tr ( " Failed to read the assets index! " ) ) ;
2013-12-10 07:12:52 +01:00
}
2014-01-14 01:13:35 +01:00
2013-12-10 07:12:52 +01:00
QList < Md5EtagDownloadPtr > dls ;
for ( auto object : index . objects . values ( ) )
{
QString objectName = object . hash . left ( 2 ) + " / " + object . hash ;
QFileInfo objectFile ( " assets/objects/ " + objectName ) ;
if ( ( ! objectFile . isFile ( ) ) | | ( objectFile . size ( ) ! = object . size ) )
{
auto objectDL = MD5EtagDownload : : make (
2013-12-13 14:58:11 +00:00
QUrl ( " http:// " + URLConstants : : RESOURCE_BASE + objectName ) ,
2013-12-10 07:12:52 +01:00
objectFile . filePath ( ) ) ;
2013-12-15 15:00:09 +01:00
objectDL - > m_total_progress = object . size ;
2013-12-10 07:12:52 +01:00
dls . append ( objectDL ) ;
}
}
2014-01-14 01:13:35 +01:00
if ( dls . size ( ) )
2013-12-10 07:12:52 +01:00
{
2013-12-23 15:46:01 +00:00
setStatus ( tr ( " Getting the assets files from Mojang... " ) ) ;
2014-03-23 19:07:13 +01:00
auto job = new NetJob ( tr ( " Assets for %1 " ) . arg ( inst - > name ( ) ) ) ;
2014-01-14 01:13:35 +01:00
for ( auto dl : dls )
2013-12-10 07:12:52 +01:00
job - > addNetAction ( dl ) ;
jarlibDownloadJob . reset ( job ) ;
connect ( jarlibDownloadJob . get ( ) , SIGNAL ( succeeded ( ) ) , SLOT ( assetsFinished ( ) ) ) ;
connect ( jarlibDownloadJob . get ( ) , SIGNAL ( failed ( ) ) , SLOT ( assetsFailed ( ) ) ) ;
connect ( jarlibDownloadJob . get ( ) , SIGNAL ( progress ( qint64 , qint64 ) ) ,
2014-01-14 01:13:35 +01:00
SIGNAL ( progress ( qint64 , qint64 ) ) ) ;
2013-12-10 07:12:52 +01:00
jarlibDownloadJob - > start ( ) ;
return ;
}
assetsFinished ( ) ;
}
void OneSixUpdate : : assetIndexFailed ( )
{
2014-03-09 23:42:25 +01:00
emitFailed ( tr ( " Failed to download the assets index! " ) ) ;
2013-12-10 07:12:52 +01:00
}
void OneSixUpdate : : assetsFinished ( )
{
2014-01-27 03:00:49 +01:00
emitSucceeded ( ) ;
2013-12-10 07:12:52 +01:00
}
void OneSixUpdate : : assetsFailed ( )
{
2014-03-09 23:42:25 +01:00
emitFailed ( tr ( " Failed to download assets! " ) ) ;
2013-12-10 07:12:52 +01:00
}
2013-08-05 03:29:50 +02:00
void OneSixUpdate : : jarlibStart ( )
{
2013-12-23 15:46:01 +00:00
setStatus ( tr ( " Getting the library files from Mojang... " ) ) ;
2013-11-25 00:46:52 +01:00
QLOG_INFO ( ) < < m_inst - > name ( ) < < " : downloading libraries " ;
2013-09-22 04:21:36 +02:00
OneSixInstance * inst = ( OneSixInstance * ) m_inst ;
2014-03-09 23:42:25 +01:00
try
{
inst - > reloadVersion ( ) ;
}
catch ( MMCError & e )
{
emitFailed ( e . cause ( ) ) ;
return ;
}
catch ( . . . )
2013-08-05 03:29:50 +02:00
{
2014-03-09 23:42:25 +01:00
emitFailed ( tr ( " Failed to load the version description file for reasons unknown. " ) ) ;
2013-08-05 03:29:50 +02:00
return ;
}
2013-09-22 04:21:36 +02:00
2013-12-10 20:33:24 +01:00
// Build a list of URLs that will need to be downloaded.
2014-03-01 23:06:47 +01:00
std : : shared_ptr < VersionFinal > version = inst - > getFullVersion ( ) ;
2013-12-10 20:33:24 +01:00
// minecraft.jar for this version
{
QString version_id = version - > id ;
QString localPath = version_id + " / " + version_id + " .jar " ;
2013-12-13 14:58:11 +00:00
QString urlstr = " http:// " + URLConstants : : AWS_DOWNLOAD_VERSIONS + localPath ;
2013-09-22 04:21:36 +02:00
2014-03-23 19:07:13 +01:00
auto job = new NetJob ( tr ( " Libraries for instance %1 " ) . arg ( inst - > name ( ) ) ) ;
2013-09-22 04:21:36 +02:00
2013-12-10 20:33:24 +01:00
auto metacache = MMC - > metacache ( ) ;
auto entry = metacache - > resolveEntry ( " versions " , localPath ) ;
job - > addNetAction ( CacheDownload : : make ( QUrl ( urlstr ) , entry ) ) ;
jarlibDownloadJob . reset ( job ) ;
}
2013-09-22 04:21:36 +02:00
2013-08-05 03:29:50 +02:00
auto libs = version - > getActiveNativeLibs ( ) ;
libs . append ( version - > getActiveNormalLibs ( ) ) ;
2013-09-22 04:21:36 +02:00
2013-09-08 02:15:20 +02:00
auto metacache = MMC - > metacache ( ) ;
2013-11-17 11:44:18 +01:00
QList < ForgeXzDownloadPtr > ForgeLibs ;
2014-04-13 23:06:28 +02:00
QList < std : : shared_ptr < OneSixLibrary > > brokenLocalLibs ;
2013-09-22 04:21:36 +02:00
for ( auto lib : libs )
2013-08-05 03:29:50 +02:00
{
2013-10-20 23:18:40 +02:00
if ( lib - > hint ( ) = = " local " )
2014-04-13 23:06:28 +02:00
{
2014-05-03 15:40:46 +02:00
if ( ! lib - > filesExist ( m_inst - > librariesPath ( ) ) )
2014-04-13 23:06:28 +02:00
brokenLocalLibs . append ( lib ) ;
2013-10-20 23:18:40 +02:00
continue ;
2014-04-13 23:06:28 +02:00
}
2013-11-25 00:46:52 +01:00
2014-01-14 01:13:35 +01:00
QString raw_storage = lib - > storagePath ( ) ;
QString raw_dl = lib - > downloadUrl ( ) ;
2013-11-25 00:46:52 +01:00
2014-01-14 01:13:35 +01:00
auto f = [ & ] ( QString storage , QString dl )
2013-09-08 02:15:20 +02:00
{
2014-01-14 01:13:35 +01:00
auto entry = metacache - > resolveEntry ( " libraries " , storage ) ;
if ( entry - > stale )
2013-11-17 11:44:18 +01:00
{
2014-01-14 01:13:35 +01:00
if ( lib - > hint ( ) = = " forge-pack-xz " )
{
ForgeLibs . append ( ForgeXzDownload : : make ( storage , entry ) ) ;
}
else
{
jarlibDownloadJob - > addNetAction ( CacheDownload : : make ( dl , entry ) ) ;
}
2013-11-17 11:44:18 +01:00
}
2014-01-14 01:13:35 +01:00
} ;
if ( raw_storage . contains ( " ${arch} " ) )
{
QString cooked_storage = raw_storage ;
QString cooked_dl = raw_dl ;
f ( cooked_storage . replace ( " ${arch} " , " 32 " ) , cooked_dl . replace ( " ${arch} " , " 32 " ) ) ;
cooked_storage = raw_storage ;
cooked_dl = raw_dl ;
f ( cooked_storage . replace ( " ${arch} " , " 64 " ) , cooked_dl . replace ( " ${arch} " , " 64 " ) ) ;
}
else
{
f ( raw_storage , raw_dl ) ;
2013-09-08 02:15:20 +02:00
}
2013-08-05 03:29:50 +02:00
}
2014-04-13 23:06:28 +02:00
if ( ! brokenLocalLibs . empty ( ) )
{
jarlibDownloadJob . reset ( ) ;
QStringList failed ;
for ( auto brokenLib : brokenLocalLibs )
{
failed . append ( brokenLib - > files ( ) ) ;
}
QString failed_all = failed . join ( " \n " ) ;
emitFailed ( tr ( " Some libraries marked as 'local' are missing their jar files: \n %1 \n \n You'll have to correct this problem manually. If this is an externally tracked instance, make sure to run it at least once outside of MultiMC. " ) . arg ( failed_all ) ) ;
return ;
}
2013-11-17 11:44:18 +01:00
// TODO: think about how to propagate this from the original json file... or IF AT ALL
QString forgeMirrorList = " http://files.minecraftforge.net/mirror-brand.list " ;
if ( ! ForgeLibs . empty ( ) )
{
jarlibDownloadJob - > addNetAction (
ForgeMirrors : : make ( ForgeLibs , jarlibDownloadJob , forgeMirrorList ) ) ;
}
2013-10-06 01:13:40 +02:00
connect ( jarlibDownloadJob . get ( ) , SIGNAL ( succeeded ( ) ) , SLOT ( jarlibFinished ( ) ) ) ;
connect ( jarlibDownloadJob . get ( ) , SIGNAL ( failed ( ) ) , SLOT ( jarlibFailed ( ) ) ) ;
connect ( jarlibDownloadJob . get ( ) , SIGNAL ( progress ( qint64 , qint64 ) ) ,
2013-09-22 04:21:36 +02:00
SIGNAL ( progress ( qint64 , qint64 ) ) ) ;
2013-08-05 03:29:50 +02:00
2013-09-02 00:25:40 +02:00
jarlibDownloadJob - > start ( ) ;
2013-07-09 22:46:33 +02:00
}
2013-08-04 14:46:33 +02:00
void OneSixUpdate : : jarlibFinished ( )
2013-07-09 22:46:33 +02:00
{
2013-12-10 07:12:52 +01:00
assetIndexStart ( ) ;
2013-07-09 22:46:33 +02:00
}
2013-08-04 14:46:33 +02:00
void OneSixUpdate : : jarlibFailed ( )
2013-07-09 22:46:33 +02:00
{
2013-09-26 02:58:09 +02:00
QStringList failed = jarlibDownloadJob - > getFailedFiles ( ) ;
QString failed_all = failed . join ( " \n " ) ;
2014-03-09 23:42:25 +01:00
emitFailed ( tr ( " Failed to download the following files: \n %1 \n \n Please try again. " ) . arg ( failed_all ) ) ;
2013-07-07 23:51:26 +02:00
}