2013-09-08 02:15:20 +02:00
|
|
|
#include "MultiMC.h"
|
2013-09-06 21:58:51 +02:00
|
|
|
#include "HttpMetaCache.h"
|
|
|
|
#include <pathutils.h>
|
2013-09-08 02:15:20 +02:00
|
|
|
|
|
|
|
#include <QFileInfo>
|
2013-09-06 21:58:51 +02:00
|
|
|
#include <QFile>
|
2013-09-08 02:15:20 +02:00
|
|
|
#include <QTemporaryFile>
|
|
|
|
#include <QSaveFile>
|
|
|
|
#include <QDateTime>
|
|
|
|
#include <QCryptographicHash>
|
|
|
|
|
|
|
|
#include <QDebug>
|
|
|
|
|
|
|
|
#include <QJsonDocument>
|
|
|
|
#include <QJsonArray>
|
|
|
|
#include <QJsonObject>
|
|
|
|
|
|
|
|
QString MetaEntry::getFullPath()
|
|
|
|
{
|
|
|
|
return PathCombine(MMC->metacache()->getBasePath(base), path);
|
|
|
|
}
|
|
|
|
|
2013-09-06 21:58:51 +02:00
|
|
|
|
|
|
|
HttpMetaCache::HttpMetaCache(QString path)
|
2013-09-08 02:15:20 +02:00
|
|
|
:QObject()
|
2013-09-06 21:58:51 +02:00
|
|
|
{
|
|
|
|
m_index_file = path;
|
2013-09-08 02:15:20 +02:00
|
|
|
saveBatchingTimer.setSingleShot(true);
|
|
|
|
saveBatchingTimer.setTimerType(Qt::VeryCoarseTimer);
|
|
|
|
connect(&saveBatchingTimer,SIGNAL(timeout()),SLOT(SaveNow()));
|
2013-09-06 21:58:51 +02:00
|
|
|
}
|
2013-09-08 02:15:20 +02:00
|
|
|
|
2013-09-06 21:58:51 +02:00
|
|
|
HttpMetaCache::~HttpMetaCache()
|
|
|
|
{
|
2013-09-08 02:15:20 +02:00
|
|
|
saveBatchingTimer.stop();
|
|
|
|
SaveNow();
|
2013-09-06 21:58:51 +02:00
|
|
|
}
|
|
|
|
|
2013-09-08 02:15:20 +02:00
|
|
|
MetaEntryPtr HttpMetaCache::getEntry ( QString base, QString resource_path )
|
2013-09-06 21:58:51 +02:00
|
|
|
{
|
|
|
|
// no base. no base path. can't store
|
|
|
|
if(!m_entries.contains(base))
|
2013-09-08 02:15:20 +02:00
|
|
|
{
|
|
|
|
// TODO: log problem
|
|
|
|
return MetaEntryPtr();
|
|
|
|
}
|
|
|
|
EntryMap & map = m_entries[base];
|
|
|
|
if(map.entry_list.contains(resource_path))
|
|
|
|
{
|
|
|
|
return map.entry_list[resource_path];
|
|
|
|
}
|
|
|
|
return MetaEntryPtr();
|
|
|
|
}
|
|
|
|
|
|
|
|
MetaEntryPtr HttpMetaCache::resolveEntry ( QString base, QString resource_path, QString expected_etag )
|
|
|
|
{
|
|
|
|
auto entry = getEntry(base, resource_path);
|
|
|
|
// it's not present? generate a default stale entry
|
|
|
|
if(!entry)
|
|
|
|
{
|
|
|
|
return staleEntry(base, resource_path);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto & selected_base = m_entries[base];
|
|
|
|
QString real_path = PathCombine(selected_base.base_path, resource_path);
|
2013-09-06 21:58:51 +02:00
|
|
|
QFileInfo finfo(real_path);
|
|
|
|
|
2013-09-08 02:15:20 +02:00
|
|
|
// is the file really there? if not -> stale
|
2013-09-06 21:58:51 +02:00
|
|
|
if(!finfo.isFile() || !finfo.isReadable())
|
|
|
|
{
|
2013-09-08 02:15:20 +02:00
|
|
|
// if the file doesn't exist, we disown the entry
|
|
|
|
selected_base.entry_list.remove(resource_path);
|
|
|
|
return staleEntry(base, resource_path);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!expected_etag.isEmpty() && expected_etag != entry->etag)
|
|
|
|
{
|
|
|
|
// if the etag doesn't match expected, we disown the entry
|
|
|
|
selected_base.entry_list.remove(resource_path);
|
|
|
|
return staleEntry(base, resource_path);
|
2013-09-06 21:58:51 +02:00
|
|
|
}
|
|
|
|
|
2013-09-08 02:15:20 +02:00
|
|
|
// if the file changed, check md5sum
|
|
|
|
qint64 file_last_changed = finfo.lastModified().toUTC().toMSecsSinceEpoch();
|
|
|
|
if(file_last_changed != entry->last_changed_timestamp)
|
|
|
|
{
|
|
|
|
QFile input(real_path);
|
|
|
|
input.open(QIODevice::ReadOnly);
|
|
|
|
QString md5sum = QCryptographicHash::hash(input.readAll(), QCryptographicHash::Md5).toHex().constData();
|
|
|
|
if(entry->md5sum != md5sum)
|
|
|
|
{
|
|
|
|
selected_base.entry_list.remove(resource_path);
|
|
|
|
return staleEntry(base, resource_path);
|
|
|
|
}
|
|
|
|
// md5sums matched... keep entry and save the new state to file
|
|
|
|
entry->last_changed_timestamp = file_last_changed;
|
|
|
|
SaveEventually();
|
|
|
|
}
|
|
|
|
|
|
|
|
// entry passed all the checks we cared about.
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool HttpMetaCache::updateEntry ( MetaEntryPtr stale_entry )
|
|
|
|
{
|
|
|
|
if(!m_entries.contains(stale_entry->base))
|
|
|
|
{
|
|
|
|
qDebug() << "Cannot add entry with unknown base: " << stale_entry->base.toLocal8Bit();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if(stale_entry->stale)
|
|
|
|
{
|
|
|
|
qDebug() << "Cannot add stale entry: " << stale_entry->getFullPath().toLocal8Bit();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
m_entries[stale_entry->base].entry_list[stale_entry->path] = stale_entry;
|
|
|
|
SaveEventually();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
MetaEntryPtr HttpMetaCache::staleEntry(QString base, QString resource_path)
|
|
|
|
{
|
|
|
|
auto foo = new MetaEntry;
|
|
|
|
foo->base = base;
|
|
|
|
foo->path = resource_path;
|
|
|
|
foo->stale = true;
|
|
|
|
return MetaEntryPtr(foo);
|
2013-09-06 21:58:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void HttpMetaCache::addBase ( QString base, QString base_root )
|
|
|
|
{
|
|
|
|
// TODO: report error
|
|
|
|
if(m_entries.contains(base))
|
|
|
|
return;
|
|
|
|
// TODO: check if the base path is valid
|
|
|
|
EntryMap foo;
|
|
|
|
foo.base_path = base_root;
|
|
|
|
m_entries[base] = foo;
|
|
|
|
}
|
|
|
|
|
2013-09-08 02:15:20 +02:00
|
|
|
QString HttpMetaCache::getBasePath ( QString base )
|
|
|
|
{
|
|
|
|
if(m_entries.contains(base))
|
|
|
|
{
|
|
|
|
return m_entries[base].base_path;
|
|
|
|
}
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-06 21:58:51 +02:00
|
|
|
void HttpMetaCache::Load()
|
|
|
|
{
|
|
|
|
QFile index(m_index_file);
|
|
|
|
if(!index.open(QIODevice::ReadOnly))
|
|
|
|
return;
|
|
|
|
|
|
|
|
QJsonDocument json = QJsonDocument::fromJson(index.readAll());
|
|
|
|
if(!json.isObject())
|
|
|
|
return;
|
|
|
|
auto root = json.object();
|
|
|
|
// check file version first
|
|
|
|
auto version_val =root.value("version");
|
|
|
|
if(!version_val.isString())
|
|
|
|
return;
|
|
|
|
if(version_val.toString() != "1")
|
|
|
|
return;
|
|
|
|
|
|
|
|
// read the entry array
|
|
|
|
auto entries_val =root.value("entries");
|
2013-09-08 02:15:20 +02:00
|
|
|
if(!entries_val.isArray())
|
2013-09-06 21:58:51 +02:00
|
|
|
return;
|
2013-09-08 02:15:20 +02:00
|
|
|
QJsonArray array = entries_val.toArray();
|
2013-09-06 21:58:51 +02:00
|
|
|
for(auto element: array)
|
|
|
|
{
|
2013-09-08 02:15:20 +02:00
|
|
|
if(!element.isObject())
|
2013-09-06 21:58:51 +02:00
|
|
|
return;
|
|
|
|
auto element_obj = element.toObject();
|
|
|
|
QString base = element_obj.value("base").toString();
|
|
|
|
if(!m_entries.contains(base))
|
|
|
|
continue;
|
|
|
|
auto & entrymap = m_entries[base];
|
|
|
|
auto foo = new MetaEntry;
|
|
|
|
foo->base = base;
|
|
|
|
QString path = foo->path = element_obj.value("path").toString();
|
|
|
|
foo->md5sum = element_obj.value("md5sum").toString();
|
|
|
|
foo->etag = element_obj.value("etag").toString();
|
|
|
|
foo->last_changed_timestamp = element_obj.value("last_changed_timestamp").toDouble();
|
2013-09-08 02:15:20 +02:00
|
|
|
// presumed innocent until closer examination
|
|
|
|
foo->stale = false;
|
2013-09-06 21:58:51 +02:00
|
|
|
entrymap.entry_list[path] = MetaEntryPtr( foo );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-08 15:59:50 +02:00
|
|
|
void HttpMetaCache::SaveEventually()
|
|
|
|
{
|
|
|
|
// reset the save timer
|
|
|
|
saveBatchingTimer.stop();
|
|
|
|
saveBatchingTimer.start(30000);
|
|
|
|
}
|
|
|
|
|
2013-09-08 02:15:20 +02:00
|
|
|
void HttpMetaCache::SaveNow()
|
2013-09-06 21:58:51 +02:00
|
|
|
{
|
|
|
|
QSaveFile tfile(m_index_file);
|
|
|
|
if(!tfile.open(QIODevice::WriteOnly | QIODevice::Truncate))
|
|
|
|
return;
|
|
|
|
QJsonObject toplevel;
|
|
|
|
toplevel.insert("version",QJsonValue(QString("1")));
|
|
|
|
QJsonArray entriesArr;
|
|
|
|
for(auto group : m_entries)
|
|
|
|
{
|
|
|
|
for(auto entry : group.entry_list)
|
|
|
|
{
|
|
|
|
QJsonObject entryObj;
|
|
|
|
entryObj.insert("base", QJsonValue(entry->base));
|
|
|
|
entryObj.insert("path", QJsonValue(entry->path));
|
|
|
|
entryObj.insert("md5sum", QJsonValue(entry->md5sum));
|
|
|
|
entryObj.insert("etag", QJsonValue(entry->etag));
|
|
|
|
entryObj.insert("last_changed_timestamp", QJsonValue(double(entry->last_changed_timestamp)));
|
|
|
|
entriesArr.append(entryObj);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
toplevel.insert("entries",entriesArr);
|
|
|
|
QJsonDocument doc(toplevel);
|
|
|
|
QByteArray jsonData = doc.toJson();
|
|
|
|
qint64 result = tfile.write(jsonData);
|
|
|
|
if(result == -1)
|
|
|
|
return;
|
|
|
|
if(result != jsonData.size())
|
|
|
|
return;
|
|
|
|
tfile.commit();
|
|
|
|
}
|