2013-08-03 23:58:39 +02:00
|
|
|
#include <QString>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QtXml/QtXml>
|
|
|
|
#include "OneSixAssets.h"
|
2013-08-07 01:38:18 +02:00
|
|
|
#include "net/DownloadJob.h"
|
2013-09-08 02:15:20 +02:00
|
|
|
#include "net/HttpMetaCache.h"
|
|
|
|
#include "MultiMC.h"
|
2013-08-03 23:58:39 +02:00
|
|
|
|
|
|
|
inline QDomElement getDomElementByTagName(QDomElement parent, QString tagname)
|
|
|
|
{
|
|
|
|
QDomNodeList elementList = parent.elementsByTagName(tagname);
|
|
|
|
if (elementList.count())
|
|
|
|
return elementList.at(0).toElement();
|
|
|
|
else
|
|
|
|
return QDomElement();
|
|
|
|
}
|
|
|
|
|
|
|
|
class ThreadedDeleter : public QThread
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
public:
|
|
|
|
void run()
|
|
|
|
{
|
2013-09-02 00:25:40 +02:00
|
|
|
QDirIterator iter ( m_base, QDirIterator::Subdirectories );
|
2013-08-03 23:58:39 +02:00
|
|
|
int base_length = m_base.length();
|
2013-09-02 00:25:40 +02:00
|
|
|
while ( iter.hasNext() )
|
2013-08-03 23:58:39 +02:00
|
|
|
{
|
|
|
|
QString filename = iter.next();
|
2013-09-02 00:25:40 +02:00
|
|
|
QFileInfo current ( filename );
|
2013-08-03 23:58:39 +02:00
|
|
|
// we keep the dirs... whatever
|
2013-09-02 00:25:40 +02:00
|
|
|
if ( current.isDir() )
|
2013-08-03 23:58:39 +02:00
|
|
|
continue;
|
|
|
|
QString trimmedf = filename;
|
2013-09-02 00:25:40 +02:00
|
|
|
trimmedf.remove ( 0, base_length + 1 );
|
|
|
|
if ( m_whitelist.contains ( trimmedf ) )
|
2013-08-03 23:58:39 +02:00
|
|
|
{
|
2013-08-04 04:19:10 +02:00
|
|
|
// qDebug() << trimmedf << " gets to live";
|
2013-08-03 23:58:39 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// DO NOT TOLERATE JUNK
|
2013-08-04 04:19:10 +02:00
|
|
|
// qDebug() << trimmedf << " dies";
|
2013-09-02 00:25:40 +02:00
|
|
|
QFile f ( filename );
|
2013-08-03 23:58:39 +02:00
|
|
|
f.remove();
|
|
|
|
}
|
|
|
|
}
|
2013-09-02 00:25:40 +02:00
|
|
|
}
|
2013-08-03 23:58:39 +02:00
|
|
|
QString m_base;
|
|
|
|
QStringList m_whitelist;
|
|
|
|
};
|
|
|
|
|
2013-09-02 00:25:40 +02:00
|
|
|
void OneSixAssets::downloadFinished()
|
2013-08-03 23:58:39 +02:00
|
|
|
{
|
2013-09-02 00:25:40 +02:00
|
|
|
deleter = new ThreadedDeleter();
|
|
|
|
QDir dir("assets");
|
|
|
|
deleter->m_base = dir.absolutePath();
|
|
|
|
deleter->m_whitelist = nuke_whitelist;
|
|
|
|
connect(deleter, SIGNAL(finished()), SIGNAL(finished()));
|
|
|
|
deleter->start();
|
|
|
|
}
|
2013-08-03 23:58:39 +02:00
|
|
|
|
|
|
|
|
2013-09-02 00:25:40 +02:00
|
|
|
void OneSixAssets::fetchXMLFinished()
|
2013-08-03 23:58:39 +02:00
|
|
|
{
|
|
|
|
QString prefix ( "http://s3.amazonaws.com/Minecraft.Resources/" );
|
|
|
|
QString fprefix ( "assets/" );
|
2013-09-02 00:25:40 +02:00
|
|
|
nuke_whitelist.clear();
|
2013-08-03 23:58:39 +02:00
|
|
|
|
2013-09-02 00:25:40 +02:00
|
|
|
auto firstJob = index_job->first();
|
2013-09-08 02:15:20 +02:00
|
|
|
QByteArray ba = firstJob.dynamicCast<ByteArrayDownload>()->m_data;
|
2013-08-03 23:58:39 +02:00
|
|
|
|
|
|
|
QString xmlErrorMsg;
|
|
|
|
QDomDocument doc;
|
|
|
|
if ( !doc.setContent ( ba, false, &xmlErrorMsg ) )
|
|
|
|
{
|
2013-09-02 00:25:40 +02:00
|
|
|
qDebug() << "Failed to process s3.amazonaws.com/Minecraft.Resources. XML error:" << xmlErrorMsg << ba;
|
2013-08-03 23:58:39 +02:00
|
|
|
}
|
|
|
|
//QRegExp etag_match(".*([a-f0-9]{32}).*");
|
|
|
|
QDomNodeList contents = doc.elementsByTagName ( "Contents" );
|
|
|
|
|
2013-09-08 02:15:20 +02:00
|
|
|
DownloadJob *job = new DownloadJob("Assets");
|
2013-09-02 00:25:40 +02:00
|
|
|
connect ( job, SIGNAL(succeeded()), SLOT(downloadFinished()) );
|
|
|
|
connect ( job, SIGNAL(failed()), SIGNAL(failed()) );
|
2013-08-03 23:58:39 +02:00
|
|
|
|
2013-09-08 02:15:20 +02:00
|
|
|
auto metacache = MMC->metacache();
|
|
|
|
|
2013-08-03 23:58:39 +02:00
|
|
|
for ( int i = 0; i < contents.length(); i++ )
|
|
|
|
{
|
|
|
|
QDomElement element = contents.at ( i ).toElement();
|
|
|
|
|
|
|
|
if ( element.isNull() )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
QDomElement keyElement = getDomElementByTagName ( element, "Key" );
|
|
|
|
QDomElement lastmodElement = getDomElementByTagName ( element, "LastModified" );
|
|
|
|
QDomElement etagElement = getDomElementByTagName ( element, "ETag" );
|
|
|
|
QDomElement sizeElement = getDomElementByTagName ( element, "Size" );
|
|
|
|
|
|
|
|
if ( keyElement.isNull() || lastmodElement.isNull() || etagElement.isNull() || sizeElement.isNull() )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
QString keyStr = keyElement.text();
|
|
|
|
QString lastModStr = lastmodElement.text();
|
|
|
|
QString etagStr = etagElement.text();
|
|
|
|
QString sizeStr = sizeElement.text();
|
|
|
|
|
|
|
|
//Filter folder keys
|
|
|
|
if ( sizeStr == "0" )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
nuke_whitelist.append ( keyStr );
|
2013-09-08 02:15:20 +02:00
|
|
|
|
|
|
|
auto entry = metacache->resolveEntry("assets", keyStr, etagStr);
|
|
|
|
if(entry->stale)
|
2013-09-02 00:25:40 +02:00
|
|
|
{
|
2013-09-30 02:34:46 +02:00
|
|
|
job->addCacheDownload(QUrl(prefix + keyStr), entry);
|
2013-09-02 00:25:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if(job->size())
|
|
|
|
{
|
|
|
|
files_job.reset ( job );
|
|
|
|
files_job->start();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
delete job;
|
|
|
|
emit finished();
|
2013-08-03 23:58:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
void OneSixAssets::start()
|
|
|
|
{
|
2013-09-08 02:15:20 +02:00
|
|
|
auto job = new DownloadJob("Assets index");
|
2013-09-30 02:34:46 +02:00
|
|
|
job->addByteArrayDownload(QUrl ( "http://s3.amazonaws.com/Minecraft.Resources/" ));
|
2013-09-02 00:25:40 +02:00
|
|
|
connect ( job, SIGNAL(succeeded()), SLOT ( fetchXMLFinished() ) );
|
2013-08-04 04:19:10 +02:00
|
|
|
index_job.reset ( job );
|
2013-09-02 00:25:40 +02:00
|
|
|
job->start();
|
2013-08-03 23:58:39 +02:00
|
|
|
}
|
|
|
|
|
2013-09-02 00:25:40 +02:00
|
|
|
|
|
|
|
#include "OneSixAssets.moc"
|