2022-07-24 15:11:41 -03:00
|
|
|
#include "HashUtils.h"
|
|
|
|
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QFile>
|
|
|
|
|
|
|
|
#include "FileSystem.h"
|
2022-11-07 14:04:48 -03:00
|
|
|
#include "StringUtils.h"
|
2022-07-24 15:11:41 -03:00
|
|
|
|
|
|
|
#include <MurmurHash2.h>
|
|
|
|
|
|
|
|
namespace Hashing {
|
|
|
|
|
|
|
|
static ModPlatform::ProviderCapabilities ProviderCaps;
|
|
|
|
|
2022-11-25 09:23:46 -03:00
|
|
|
Hasher::Ptr createHasher(QString file_path, ModPlatform::ResourceProvider provider)
|
2022-07-24 15:11:41 -03:00
|
|
|
{
|
|
|
|
switch (provider) {
|
2022-11-25 09:23:46 -03:00
|
|
|
case ModPlatform::ResourceProvider::MODRINTH:
|
2022-07-24 15:11:41 -03:00
|
|
|
return createModrinthHasher(file_path);
|
2022-11-25 09:23:46 -03:00
|
|
|
case ModPlatform::ResourceProvider::FLAME:
|
2022-07-24 15:11:41 -03:00
|
|
|
return createFlameHasher(file_path);
|
|
|
|
default:
|
|
|
|
qCritical() << "[Hashing]"
|
|
|
|
<< "Unrecognized mod platform!";
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Hasher::Ptr createModrinthHasher(QString file_path)
|
|
|
|
{
|
2023-01-24 16:52:09 -03:00
|
|
|
return makeShared<ModrinthHasher>(file_path);
|
2022-07-24 15:11:41 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
Hasher::Ptr createFlameHasher(QString file_path)
|
|
|
|
{
|
2023-01-24 16:52:09 -03:00
|
|
|
return makeShared<FlameHasher>(file_path);
|
2022-07-24 15:11:41 -03:00
|
|
|
}
|
|
|
|
|
2022-11-25 09:23:46 -03:00
|
|
|
Hasher::Ptr createBlockedModHasher(QString file_path, ModPlatform::ResourceProvider provider)
|
2022-10-25 01:19:19 -07:00
|
|
|
{
|
2023-01-24 16:52:09 -03:00
|
|
|
return makeShared<BlockedModHasher>(file_path, provider);
|
2022-10-25 01:19:19 -07:00
|
|
|
}
|
|
|
|
|
2022-11-25 09:23:46 -03:00
|
|
|
Hasher::Ptr createBlockedModHasher(QString file_path, ModPlatform::ResourceProvider provider, QString type)
|
2022-10-25 01:19:19 -07:00
|
|
|
{
|
2023-01-24 16:52:09 -03:00
|
|
|
auto hasher = makeShared<BlockedModHasher>(file_path, provider);
|
2022-10-25 01:19:19 -07:00
|
|
|
hasher->useHashType(type);
|
|
|
|
return hasher;
|
|
|
|
}
|
|
|
|
|
2022-07-24 15:11:41 -03:00
|
|
|
void ModrinthHasher::executeTask()
|
|
|
|
{
|
|
|
|
QFile file(m_path);
|
|
|
|
|
|
|
|
try {
|
|
|
|
file.open(QFile::ReadOnly);
|
|
|
|
} catch (FS::FileSystemException& e) {
|
|
|
|
qCritical() << QString("Failed to open JAR file in %1").arg(m_path);
|
|
|
|
qCritical() << QString("Reason: ") << e.cause();
|
|
|
|
|
|
|
|
emitFailed("Failed to open file for hashing.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-11-25 09:23:46 -03:00
|
|
|
auto hash_type = ProviderCaps.hashType(ModPlatform::ResourceProvider::MODRINTH).first();
|
|
|
|
m_hash = ProviderCaps.hash(ModPlatform::ResourceProvider::MODRINTH, &file, hash_type);
|
2022-07-24 15:11:41 -03:00
|
|
|
|
|
|
|
file.close();
|
|
|
|
|
|
|
|
if (m_hash.isEmpty()) {
|
|
|
|
emitFailed("Empty hash!");
|
|
|
|
} else {
|
|
|
|
emitSucceeded();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FlameHasher::executeTask()
|
|
|
|
{
|
|
|
|
// CF-specific
|
|
|
|
auto should_filter_out = [](char c) { return (c == 9 || c == 10 || c == 13 || c == 32); };
|
|
|
|
|
2023-03-12 18:32:42 -04:00
|
|
|
std::ifstream file_stream(StringUtils::toStdString(m_path).c_str(), std::ifstream::binary);
|
2022-07-24 15:11:41 -03:00
|
|
|
// TODO: This is very heavy work, but apparently QtConcurrent can't use move semantics, so we can't boop this to another thread.
|
|
|
|
// How do we make this non-blocking then?
|
|
|
|
m_hash = QString::number(MurmurHash2(std::move(file_stream), 4 * MiB, should_filter_out));
|
|
|
|
|
|
|
|
if (m_hash.isEmpty()) {
|
|
|
|
emitFailed("Empty hash!");
|
|
|
|
} else {
|
|
|
|
emitSucceeded();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-25 01:19:19 -07:00
|
|
|
|
2022-11-25 09:23:46 -03:00
|
|
|
BlockedModHasher::BlockedModHasher(QString file_path, ModPlatform::ResourceProvider provider)
|
2022-10-25 01:19:19 -07:00
|
|
|
: Hasher(file_path), provider(provider) {
|
|
|
|
setObjectName(QString("BlockedModHasher: %1").arg(file_path));
|
|
|
|
hash_type = ProviderCaps.hashType(provider).first();
|
|
|
|
}
|
|
|
|
|
|
|
|
void BlockedModHasher::executeTask()
|
|
|
|
{
|
|
|
|
QFile file(m_path);
|
|
|
|
|
|
|
|
try {
|
|
|
|
file.open(QFile::ReadOnly);
|
|
|
|
} catch (FS::FileSystemException& e) {
|
|
|
|
qCritical() << QString("Failed to open JAR file in %1").arg(m_path);
|
|
|
|
qCritical() << QString("Reason: ") << e.cause();
|
|
|
|
|
|
|
|
emitFailed("Failed to open file for hashing.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_hash = ProviderCaps.hash(provider, &file, hash_type);
|
|
|
|
|
|
|
|
file.close();
|
|
|
|
|
|
|
|
if (m_hash.isEmpty()) {
|
|
|
|
emitFailed("Empty hash!");
|
|
|
|
} else {
|
|
|
|
emitSucceeded();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList BlockedModHasher::getHashTypes() {
|
|
|
|
return ProviderCaps.hashType(provider);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool BlockedModHasher::useHashType(QString type) {
|
|
|
|
auto types = ProviderCaps.hashType(provider);
|
|
|
|
if (types.contains(type)) {
|
|
|
|
hash_type = type;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
qDebug() << "Bad hash type " << type << " for provider";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-07-24 15:11:41 -03:00
|
|
|
} // namespace Hashing
|