MultiMC5/asset_test.cpp

155 lines
3.8 KiB
C++
Raw Normal View History

#include <QString>
#include <QDebug>
#include <QtXml/QtXml>
2013-07-05 23:50:07 +00:00
#include "dlqueue.h"
inline QDomElement getDomElementByTagName(QDomElement parent, QString tagname)
{
QDomNodeList elementList = parent.elementsByTagName(tagname);
if (elementList.count())
return elementList.at(0).toElement();
else
return QDomElement();
}
// a job that removes all files from the base folder that don't match the whitelist
// runs in whatever thread owns the queue. it is fast though.
class NukeAndPaveJob: public Job
{
public:
explicit NukeAndPaveJob(QString base, QStringList whitelist)
:Job()
{
QDir dir(base);
m_base = dir.absolutePath();
m_whitelist = whitelist;
};
virtual void start()
{
QDirIterator iter(m_base, QDirIterator::Subdirectories);
QStringList nuke_list;
int base_length = m_base.length();
while (iter.hasNext())
{
QString filename = iter.next();
QFileInfo current(filename);
// we keep the dirs... whatever
if(current.isDir())
continue;
QString trimmedf = filename;
trimmedf.remove(0, base_length + 1);
if(m_whitelist.contains(trimmedf))
{
//qDebug() << trimmedf << " gets to live";
}
else
{
// DO NOT TOLERATE JUNK
//qDebug() << trimmedf << " dies";
QFile f (filename);
f.remove();
}
}
emit finish();
};
private:
QString m_base;
QStringList m_whitelist;
};
class DlMachine : public QObject
{
Q_OBJECT
public slots:
2013-07-04 22:57:20 +00:00
void filesFinished()
{
qApp->quit();
}
void fetchFinished()
{
QString prefix("http://s3.amazonaws.com/Minecraft.Resources/");
QString fprefix("assets/");
QStringList nuke_whitelist;
JobPtr firstJob = index_job->getFirstJob();
auto DlJob = firstJob.dynamicCast<DownloadJob>();
QByteArray ba = DlJob->m_data;
QString xmlErrorMsg;
QDomDocument doc;
if (!doc.setContent(ba, false, &xmlErrorMsg))
{
qDebug() << "Failed to process s3.amazonaws.com/Minecraft.Resources. XML error:" <<
xmlErrorMsg << ba;
}
QRegExp etag_match(".*([a-f0-9]{32}).*");
QDomNodeList contents = doc.elementsByTagName("Contents");
JobList *job = new JobList();
2013-07-04 22:57:20 +00:00
connect(job, SIGNAL(finished()), SLOT(filesFinished()));
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;
2013-07-04 22:57:20 +00:00
QString trimmedEtag = etagStr.remove('"');
job->add(DownloadJob::create(QUrl(prefix + keyStr),fprefix + keyStr, trimmedEtag));
nuke_whitelist.append(keyStr);
}
job->add(JobPtr(new NukeAndPaveJob(fprefix, nuke_whitelist)));
2013-07-04 22:57:20 +00:00
files_job.reset(job);
dl.enqueue(files_job);
}
void fetchStarted()
{
qDebug() << "Started downloading!";
}
public:
void start()
{
JobList *job = new JobList();
job->add(DownloadJob::create(QUrl("http://s3.amazonaws.com/Minecraft.Resources/")));
connect(job, SIGNAL(finished()), SLOT(fetchFinished()));
connect(job, SIGNAL(started()), SLOT(fetchStarted()));
2013-07-04 22:57:20 +00:00
index_job.reset(job);
dl.enqueue(index_job);
}
JobListQueue dl;
JobListPtr index_job;
JobListPtr files_job;
};
int main(int argc, char *argv[])
{
QCoreApplication app(argc, argv);
DlMachine dl;
dl.start();
return app.exec();
}
2013-07-04 22:57:20 +00:00
#include "asset_test.moc"