refactoring hwid

This commit is contained in:
0-Kutya-0 2025-09-20 21:42:46 +03:00
parent 2f3fa2c38f
commit 1ec5509c15
5 changed files with 218 additions and 92 deletions

View File

@ -0,0 +1,12 @@
#pragma once
#include <QString>
struct DeviceDetails {
QString hwid;
QString os;
QString osVersion;
QString model;
};
DeviceDetails GetDeviceDetails();

View File

@ -554,7 +554,7 @@
<item row="5" column="1">
<widget class="QCheckBox" name="sub_send_hwid">
<property name="text">
<string>Includes sending HWID, model, and OS version</string>
<string>Enable sending HWID, device model, and OS version when updating subscription</string>
</property>
</widget>
</item>

View File

@ -90,6 +90,10 @@
<message>
<source>Clear servers before updating subscription</source>
<translation>Очищать список серверов перед обновлением подписки</translation>
</message>
<message>
<source>Enable sending HWID, device model, and OS version when updating subscription</source>
<translation>Включить отправку HWID, модели устройства и версии ОС при обновлении подписки</translation>
</message>
<message>
<source>Core</source>

View File

@ -0,0 +1,200 @@
#include "include/global/DeviceDetailsHelper.hpp"
#include <QString>
#include <QSysInfo>
#include <QFile>
#include <vector>
#include <string>
#ifdef Q_OS_WIN
#include <windows.h>
#include "include/sys/windows/WinVersion.h"
#include <include/sys/Process.hpp>
#include <comdef.h>
#include <Wbemidl.h>
#pragma comment(lib, "wbemuuid.lib")
#endif
#ifdef Q_OS_WIN
static QString readRegistryStringValue(const QString& keyPath, const QString& valueName) {
HKEY hKey = nullptr;
std::wstring keyPathW = keyPath.toStdWString();
LONG res = RegOpenKeyExW(HKEY_LOCAL_MACHINE, keyPathW.c_str(), 0, KEY_READ | KEY_WOW64_64KEY, &hKey);
if (res != ERROR_SUCCESS) return QString();
DWORD type = 0;
DWORD cbData = 0;
res = RegQueryValueExW(hKey, (LPCWSTR)valueName.utf16(), nullptr, &type, nullptr, &cbData);
if (res != ERROR_SUCCESS || type != REG_SZ) {
RegCloseKey(hKey);
return QString();
}
std::vector<wchar_t> buffer(cbData / sizeof(wchar_t) + 1);
res = RegQueryValueExW(hKey, (LPCWSTR)valueName.utf16(), nullptr, nullptr, reinterpret_cast<LPBYTE>(buffer.data()), &cbData);
if (res != ERROR_SUCCESS) {
RegCloseKey(hKey);
return QString();
}
RegCloseKey(hKey);
return QString::fromWCharArray(buffer.data());
}
static QString queryWmiProperty(const QString& wmiClass, const QString& property) {
HRESULT hres;
hres = CoInitializeEx(0, COINIT_MULTITHREADED);
if (FAILED(hres)) return QString();
hres = CoInitializeSecurity(
NULL, -1, NULL, NULL,
RPC_C_AUTHN_LEVEL_DEFAULT,
RPC_C_IMP_LEVEL_IMPERSONATE,
NULL, EOAC_NONE, NULL
);
if (FAILED(hres) && hres != RPC_E_TOO_LATE) {
CoUninitialize();
return QString();
}
IWbemLocator* pLoc = NULL;
hres = CoCreateInstance(
CLSID_WbemLocator, 0,
CLSCTX_INPROC_SERVER,
IID_IWbemLocator, (LPVOID*)&pLoc
);
if (FAILED(hres)) {
CoUninitialize();
return QString();
}
IWbemServices* pSvc = NULL;
hres = pLoc->ConnectServer(
_bstr_t(L"ROOT\\CIMV2"),
NULL, NULL, 0, NULL, 0, 0, &pSvc
);
if (FAILED(hres)) {
pLoc->Release();
CoUninitialize();
return QString();
}
hres = CoSetProxyBlanket(
pSvc,
RPC_C_AUTHN_WINNT,
RPC_C_AUTHZ_NONE,
NULL,
RPC_C_AUTHN_LEVEL_CALL,
RPC_C_IMP_LEVEL_IMPERSONATE,
NULL,
EOAC_NONE
);
if (FAILED(hres)) {
pSvc->Release();
pLoc->Release();
CoUninitialize();
return QString();
}
IEnumWbemClassObject* pEnumerator = NULL;
QString query = QString("SELECT %1 FROM %2").arg(property, wmiClass);
hres = pSvc->ExecQuery(
bstr_t("WQL"),
bstr_t(query.toStdWString().c_str()),
WBEM_FLAG_FORWARD_ONLY | WBEM_FLAG_RETURN_IMMEDIATELY,
NULL,
&pEnumerator
);
if (FAILED(hres)) {
pSvc->Release();
pLoc->Release();
CoUninitialize();
return QString();
}
IWbemClassObject* pclsObj = NULL;
ULONG uReturn = 0;
QString result;
if (pEnumerator) {
HRESULT hr = pEnumerator->Next(WBEM_INFINITE, 1, &pclsObj, &uReturn);
if (uReturn) {
VARIANT vtProp;
VariantInit(&vtProp);
hr = pclsObj->Get(property.toStdWString().c_str(), 0, &vtProp, 0, 0);
if (SUCCEEDED(hr) && vtProp.vt == VT_BSTR) {
result = QString::fromWCharArray(vtProp.bstrVal);
}
VariantClear(&vtProp);
pclsObj->Release();
}
pEnumerator->Release();
}
pSvc->Release();
pLoc->Release();
CoUninitialize();
return result;
}
static QString winBaseBoard() {
return queryWmiProperty("Win32_BaseBoard", "Product");
}
static QString winModel() {
return queryWmiProperty("Win32_ComputerSystem", "Model");
}
#endif
DeviceDetails GetDeviceDetails() {
DeviceDetails details;
#ifdef Q_OS_WIN
const QString regPath = QStringLiteral("SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\SoftwareProtectionPlatform\\Plugins\\Objects\\msft:rm/algorithm/hwid/4.0");
const QString valueName = QStringLiteral("ModuleId");
details.hwid = readRegistryStringValue(regPath, valueName);
if (details.hwid.isEmpty()) {
auto productId = QSysInfo::machineUniqueId();
if (productId.isEmpty()) productId = QSysInfo::productType().toUtf8();
details.hwid = QString("%1-%2").arg(QSysInfo::machineHostName(), QString::fromUtf8(productId));
}
details.os = QStringLiteral("Windows");
VersionInfo info;
WinVersion::GetVersion(info);
details.osVersion = QString("%1.%2.%3").arg(info.Major).arg(info.Minor).arg(info.BuildNum);
auto wm = winModel();
auto wbb = winBaseBoard();
details.model = (wm == wbb) ? wm : wm + "/" + wbb;
#elif defined(Q_OS_LINUX)
QString mid;
QFile f1("/etc/machine-id");
if (f1.exists() && f1.open(QIODevice::ReadOnly | QIODevice::Text)) {
mid = QString::fromUtf8(f1.readAll()).trimmed();
f1.close();
}
else {
QFile f2("/var/lib/dbus/machine-id");
if (f2.exists() && f2.open(QIODevice::ReadOnly | QIODevice::Text)) {
mid = QString::fromUtf8(f2.readAll()).trimmed();
f2.close();
}
}
details.hwid = mid;
details.os = QStringLiteral("Linux");
details.osVersion = QSysInfo::kernelVersion();
details.model = QSysInfo::prettyProductName();
#elif defined(Q_OS_MACOS)
details.hwid = QSysInfo::machineUniqueId();
details.os = QStringLiteral("macOS");
details.osVersion = QSysInfo::productVersion();
details.model = QSysInfo::prettyProductName();
#else
details.hwid = QSysInfo::machineUniqueId();
details.os = QSysInfo::productType();
details.osVersion = QSysInfo::productVersion();
details.model = QSysInfo::prettyProductName();
#endif
return details;
}

View File

@ -1,9 +1,3 @@
#include <QCoreApplication>
#ifdef Q_OS_WIN
#include <windows.h>
#endif
#undef boolean
#include "include/global/HTTPRequestHelper.hpp"
#include <QNetworkProxy>
@ -16,94 +10,10 @@
#include "include/global/Configs.hpp"
#include "include/ui/mainwindow.h"
#include "include/global/DeviceDetailsHelper.hpp"
namespace Configs_network {
namespace {
struct DeviceDetails {
QString hwid;
QString os;
QString osVersion;
QString model;
};
#ifdef Q_OS_WIN
static QString readRegistryStringValue(const QString &keyPath, const QString &valueName) {
HKEY hKey = nullptr;
std::wstring keyPathW = keyPath.toStdWString();
LONG res = RegOpenKeyExW(HKEY_LOCAL_MACHINE, keyPathW.c_str(), 0, KEY_READ | KEY_WOW64_64KEY, &hKey);
if (res != ERROR_SUCCESS) return QString();
DWORD type = 0;
DWORD cbData = 0;
res = RegQueryValueExW(hKey, (LPCWSTR)valueName.utf16(), nullptr, &type, nullptr, &cbData);
if (res != ERROR_SUCCESS || type != REG_SZ) {
RegCloseKey(hKey);
return QString();
}
std::vector<wchar_t> buffer(cbData/sizeof(wchar_t) + 1);
res = RegQueryValueExW(hKey, (LPCWSTR)valueName.utf16(), nullptr, nullptr, reinterpret_cast<LPBYTE>(buffer.data()), &cbData);
if (res != ERROR_SUCCESS) {
RegCloseKey(hKey);
return QString();
}
RegCloseKey(hKey);
return QString::fromWCharArray(buffer.data());
}
#endif
static DeviceDetails GetDeviceDetails() {
DeviceDetails details;
#ifdef Q_OS_WIN
const QString regPath = QString::fromUtf8("SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\SoftwareProtectionPlatform\\Plugins\\Objects\\msft:rm/algorithm/hwid/4.0");
const QString valueName = QString::fromUtf8("ModuleId");
details.hwid = readRegistryStringValue(regPath, valueName);
if (details.hwid.isEmpty()) {
auto productId = QSysInfo::machineUniqueId();
if (productId.isEmpty()) productId = QSysInfo::productType().toUtf8();
details.hwid = QString("%1-%2").arg(QSysInfo::machineHostName(), QString::fromUtf8(productId));
}
details.os = QStringLiteral("Windows");
VersionInfo info;
WinVersion::GetVersion(info);
details.osVersion = QString("%1.%2.%3").arg(info.Major).arg(info.Minor).arg(info.BuildNum);
details.model = QSysInfo::prettyProductName();
#elif defined(Q_OS_LINUX)
QString mid;
QFile f1("/etc/machine-id");
if (f1.exists() && f1.open(QIODevice::ReadOnly | QIODevice::Text)) {
mid = QString::fromUtf8(f1.readAll()).trimmed();
f1.close();
} else {
QFile f2("/var/lib/dbus/machine-id");
if (f2.exists() && f2.open(QIODevice::ReadOnly | QIODevice::Text)) {
mid = QString::fromUtf8(f2.readAll()).trimmed();
f2.close();
}
}
details.hwid = mid;
details.os = QStringLiteral("Linux");
details.osVersion = QSysInfo::kernelVersion();
details.model = QSysInfo::prettyProductName();
#elif defined(Q_OS_MACOS)
details.hwid = QSysInfo::machineUniqueId();
details.os = QStringLiteral("macOS");
details.osVersion = QSysInfo::productVersion();
details.model = QSysInfo::prettyProductName();
#else
details.hwid = QSysInfo::machineUniqueId();
details.os = QSysInfo::productType();
details.osVersion = QSysInfo::productVersion();
details.model = QSysInfo::prettyProductName();
#endif
return details;
}
}
HTTPResponse NetworkRequestHelper::HttpGet(const QString &url) {
QNetworkRequest request;
QNetworkAccessManager accessManager;