mirror of
https://github.com/amnezia-vpn/amnezia-client.git
synced 2026-05-08 14:33:23 +00:00
* feat: initial conan support * feat: add awg-go and awg-apple recipes * feat: macos full feature conan build, except ss and cloak * feat: conan android initial support * fix: android libssh fixes * conan: android additional recipes and fixes * feat: openvpn add support android * fix: awg android connection establish * conan: apple full-featured support * chore: bump min macos version * chore: get rid of manual deploy recursive copying * conan: beautify makefile-based recipes * conan: add geosite.dat and geoip.dat * conan: use lib linking instead of QT_EXTRA_LIBS for OVPN * conan: address lack of SONAME of libck-ovpn-plugin.so correctly * conan: windows initial support * conan: make awg-windows and wintun be interpret as exes * conan: fix version for v2ray-rules-dat * feat: conan and platform bootstrap rework in cmake * feat: 16kb support for Android * chore(conan): recipes cleanup * feat: support of drivers for windows * feat: support full-featured cmake install * chore: exclude qtkeychain from the target build * fix: install for apple systems * fix: provide flags for cloak plugin for openvpn-pt-android * chore: bump android deps for 16kb support * feat(conan): patch cloak to properly provide env for golang * chore: remove redundant hint from conan find * feat: linux <-> conan features * feat: linux initial packaging support * feat: linux cpack support * feat: cpack windows full-featured build * feat: productbuild cpack support * feat: rework CI/CD for macos * feat: rework CI/CD for Linux * fix: libncap automake args * fix: CI/CD correct QT paths * fix: windows rework CI/CD * fix: windows artifact upload * chore: remove MacOS-old from build targets * feat: add conan to all mobile and NE builds * feat: support default amnezia conan remote * fix: use Release instead of release on Android * feat: get rid of 3rd-prebuilt * feat: conan CI/CD upload * fix: CI/CD change windows toolset versions * fix: remove MSVC version from CI/CD * feat: conan CI/CD add Release and Debug build types * feat: add multiple xcode versions for conan CI/CD * fix: correct conan CI/CD clang versions * feat: separate prebuilt baking, and add some for NE * feat: rework keychain on ios/macos even more * fix: add desktop Qt for iOS * feat: add QT_HOST_PATH to build.sh * fix: add deploy definition to cmake * fix: android adjustments for toolchains and CI/CD * fix: add needs for Android CI/CD * fix: Android CI/CD use android-28 * fix: modernize translations, and CI/CD fixes * fix: gradle min sdk compilation error * fix: CI/CD add installers to all jobs * fix: parse android platform more precisely * fix: adjust aab path in CI/CD * feat: CI/CD do not execute artifact build if there is nothing changed * fix: CI/CD use common jobs even if previous were failed * fix: Apple CI/CD use set-key-partition-list for keychains * fix: Apple CI/CD do not specify any keychain (use default) * fix: build aab as a different step in build script * chore: beautify build.sh script * feat: CI/CD build separate APKs per ABI * fix: Android CI/CD upload artifact in separate steps * chore: recipes cleanup * feat: add hints for conan on MacOS * fix: add main.cpp and tests back to CMakeLists.txt * chore: xrayProtocol codestyle changes * fix: openssl set proper X509 request version * fix: make openvpn protocol rely only on client while configuring * chore: get rid of old scripts * chore: readme update describing build process more precisely * feat: windows build script add multiprocessing capabilities * chore: bump Qt version in README * feat: add generator option and use Ninja by default in CI/CD for linux/macos --------- Co-authored-by: NickVs2015 <nv@amnezia.org>
345 lines
9.2 KiB
C++
Executable File
345 lines
9.2 KiB
C++
Executable File
#include <QCoreApplication>
|
|
#include <QDebug>
|
|
#include <QDir>
|
|
#include <QProcess>
|
|
#include <QRandomGenerator>
|
|
#include <QRegularExpression>
|
|
#include <QStandardPaths>
|
|
#include <QUrl>
|
|
#include <QJsonDocument>
|
|
#include <QJsonObject>
|
|
|
|
#include "core/utils/utilities.h"
|
|
|
|
#ifdef Q_OS_WINDOWS
|
|
QString printErrorMessage(DWORD errorCode) {
|
|
LPVOID lpMsgBuf;
|
|
|
|
DWORD dwFlags = FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
|
FORMAT_MESSAGE_IGNORE_INSERTS;
|
|
|
|
DWORD dwLanguageId = MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US);
|
|
|
|
FormatMessageW(
|
|
dwFlags,
|
|
NULL,
|
|
errorCode,
|
|
dwLanguageId,
|
|
(LPWSTR)&lpMsgBuf,
|
|
0,
|
|
NULL
|
|
);
|
|
|
|
QString errorMsg = QString::fromWCharArray((LPCWSTR)lpMsgBuf);
|
|
LocalFree(lpMsgBuf);
|
|
return errorMsg.trimmed();
|
|
}
|
|
|
|
QString Utils::getNextDriverLetter()
|
|
{
|
|
DWORD drivesBitmask = GetLogicalDrives();
|
|
if (drivesBitmask == 0) {
|
|
DWORD error = GetLastError();
|
|
qDebug() << "GetLogicalDrives failed. Error code:" << error;
|
|
return "";
|
|
}
|
|
|
|
QString letters = "FGHIJKLMNOPQRSTUVWXYZ";
|
|
QString availableLetter;
|
|
|
|
for (int i = letters.size() - 1; i >= 0; --i) {
|
|
QChar letterChar = letters.at(i);
|
|
int driveIndex = letterChar.toLatin1() - 'A';
|
|
|
|
if ((drivesBitmask & (1 << driveIndex)) == 0) {
|
|
availableLetter = letterChar;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (availableLetter.isEmpty()) {
|
|
qDebug() << "Can't find free drive letter";
|
|
return "";
|
|
}
|
|
|
|
return availableLetter;
|
|
}
|
|
#endif
|
|
|
|
QString Utils::getRandomString(int len)
|
|
{
|
|
const QString possibleCharacters = QStringLiteral("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789");
|
|
QString randomString;
|
|
|
|
for (int i = 0; i < len; ++i) {
|
|
randomString.append(possibleCharacters.at(QRandomGenerator::system()->bounded(possibleCharacters.length())));
|
|
}
|
|
|
|
return randomString;
|
|
}
|
|
|
|
QString Utils::VerifyJsonString(const QString &source)
|
|
{
|
|
QJsonParseError error;
|
|
QJsonDocument doc = QJsonDocument::fromJson(source.toUtf8(), &error);
|
|
Q_UNUSED(doc)
|
|
|
|
if (error.error == QJsonParseError::NoError) {
|
|
return "";
|
|
} else {
|
|
qDebug() << "WARNING: Json parse returns: " + error.errorString();
|
|
return error.errorString();
|
|
}
|
|
}
|
|
|
|
QJsonObject Utils::JsonFromString(const QString &string)
|
|
{
|
|
auto removeComment = string.trimmed();
|
|
if (removeComment != string.trimmed()) {
|
|
qDebug() << "Some comments have been removed from the json.";
|
|
}
|
|
QJsonDocument doc = QJsonDocument::fromJson(removeComment.toUtf8());
|
|
return doc.object();
|
|
}
|
|
|
|
QString Utils::SafeBase64Decode(QString string)
|
|
{
|
|
QByteArray ba = string.replace(QChar('-'), QChar('+')).replace(QChar('_'), QChar('/')).toUtf8();
|
|
return QByteArray::fromBase64(ba, QByteArray::Base64Option::OmitTrailingEquals);
|
|
}
|
|
|
|
QString Utils::JsonToString(const QJsonObject &json, QJsonDocument::JsonFormat format)
|
|
{
|
|
QJsonDocument doc;
|
|
doc.setObject(json);
|
|
return doc.toJson(format);
|
|
}
|
|
|
|
QString Utils::JsonToString(const QJsonArray &array, QJsonDocument::JsonFormat format)
|
|
{
|
|
QJsonDocument doc;
|
|
doc.setArray(array);
|
|
return doc.toJson(format);
|
|
}
|
|
|
|
bool Utils::initializePath(const QString &path)
|
|
{
|
|
QDir dir;
|
|
if (!dir.mkpath(path)) {
|
|
qWarning().noquote() << QString("Cannot initialize path: '%1'").arg(path);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool Utils::createEmptyFile(const QString &path)
|
|
{
|
|
QFile f(path);
|
|
return f.open(QIODevice::WriteOnly | QIODevice::Truncate);
|
|
}
|
|
|
|
QString Utils::executable(const QString &baseName, bool absPath)
|
|
{
|
|
QString ext;
|
|
#ifdef Q_OS_WIN
|
|
ext = ".exe";
|
|
#endif
|
|
const QString fileName = baseName + ext;
|
|
if (!absPath) {
|
|
return fileName;
|
|
}
|
|
return QCoreApplication::applicationDirPath() + "/" + fileName;
|
|
}
|
|
|
|
QString Utils::usrExecutable(const QString &baseName)
|
|
{
|
|
if (QFileInfo::exists("/usr/sbin/" + baseName))
|
|
return ("/usr/sbin/" + baseName);
|
|
else
|
|
return ("/usr/bin/" + baseName);
|
|
}
|
|
|
|
bool Utils::processIsRunning(const QString &fileName, const bool fullFlag)
|
|
{
|
|
#ifdef Q_OS_WIN
|
|
HANDLE hSnapshot = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
|
|
if (hSnapshot == INVALID_HANDLE_VALUE) {
|
|
qWarning() << "Utils::processIsRunning error CreateToolhelp32Snapshot";
|
|
return false;
|
|
}
|
|
|
|
PROCESSENTRY32W pe32;
|
|
pe32.dwSize = sizeof(PROCESSENTRY32W);
|
|
|
|
if (!Process32FirstW(hSnapshot, &pe32)) {
|
|
CloseHandle(hSnapshot);
|
|
qWarning() << "Utils::processIsRunning error Process32FirstW";
|
|
return false;
|
|
}
|
|
|
|
do {
|
|
QString exeFile = QString::fromWCharArray(pe32.szExeFile);
|
|
|
|
if (exeFile.compare(fileName, Qt::CaseInsensitive) == 0) {
|
|
CloseHandle(hSnapshot);
|
|
return true;
|
|
}
|
|
} while (Process32NextW(hSnapshot, &pe32));
|
|
|
|
CloseHandle(hSnapshot);
|
|
return false;
|
|
|
|
#elif defined(Q_OS_IOS) || defined(Q_OS_ANDROID) || defined(MACOS_NE)
|
|
return false;
|
|
#else
|
|
QProcess process;
|
|
QStringList arguments;
|
|
if (fullFlag) {
|
|
arguments << "-f";
|
|
}
|
|
arguments << fileName;
|
|
process.setProcessChannelMode(QProcess::MergedChannels);
|
|
process.start("pgrep", arguments);
|
|
process.waitForFinished();
|
|
if (process.exitStatus() == QProcess::NormalExit) {
|
|
if (fullFlag) {
|
|
return (process.readLine().toUInt() > 0);
|
|
} else {
|
|
return (process.readAll().toUInt() > 0);
|
|
}
|
|
}
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
bool Utils::killProcessByName(const QString &name)
|
|
{
|
|
qDebug().noquote() << "Kill process" << name;
|
|
#ifdef Q_OS_WIN
|
|
HANDLE hSnapshot = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
|
|
if (hSnapshot == INVALID_HANDLE_VALUE)
|
|
return false;
|
|
|
|
PROCESSENTRY32W pe32;
|
|
pe32.dwSize = sizeof(PROCESSENTRY32W);
|
|
|
|
bool success = false;
|
|
|
|
if (Process32FirstW(hSnapshot, &pe32)) {
|
|
do {
|
|
QString exeFile = QString::fromWCharArray(pe32.szExeFile);
|
|
|
|
if (exeFile.compare(name, Qt::CaseInsensitive) == 0) {
|
|
HANDLE hProcess = OpenProcess(PROCESS_TERMINATE, FALSE, pe32.th32ProcessID);
|
|
if (hProcess != NULL) {
|
|
if (TerminateProcess(hProcess, 0)) {
|
|
success = true;
|
|
} else {
|
|
DWORD error = GetLastError();
|
|
qCritical() << "Can't terminate process" << exeFile << "(PID:" << pe32.th32ProcessID << "). Error:" << printErrorMessage(error);
|
|
}
|
|
CloseHandle(hProcess);
|
|
} else {
|
|
DWORD error = GetLastError();
|
|
qCritical() << "Can't open process for termination" << exeFile << "(PID:" << pe32.th32ProcessID << "). Error:" << printErrorMessage(error);
|
|
}
|
|
}
|
|
} while (Process32NextW(hSnapshot, &pe32));
|
|
}
|
|
|
|
CloseHandle(hSnapshot);
|
|
return success;
|
|
#elif defined Q_OS_IOS || defined(Q_OS_ANDROID)
|
|
return false;
|
|
#else
|
|
return QProcess::execute("pkill", { name }) == 0;
|
|
#endif
|
|
}
|
|
|
|
QString Utils::openVpnExecPath()
|
|
{
|
|
return Utils::executable("openvpn", true);
|
|
}
|
|
|
|
QString Utils::wireguardExecPath()
|
|
{
|
|
#ifdef Q_OS_WIN
|
|
return Utils::executable("wireguard/wireguard-service", true);
|
|
#elif defined Q_OS_LINUX
|
|
return Utils::usrExecutable("wg-quick");
|
|
#elif defined Q_OS_MAC
|
|
return Utils::executable("/wireguard", true);
|
|
#else
|
|
return {};
|
|
#endif
|
|
}
|
|
|
|
QString Utils::certUtilPath()
|
|
{
|
|
#ifdef Q_OS_WIN
|
|
QString winPath = QString::fromUtf8(qgetenv("windir"));
|
|
return winPath + "\\system32\\certutil.exe";
|
|
#else
|
|
return "";
|
|
#endif
|
|
}
|
|
|
|
QString Utils::tun2socksPath()
|
|
{
|
|
return Utils::executable("tun2socks", true);
|
|
}
|
|
|
|
#ifdef Q_OS_WIN
|
|
// Inspired from http://stackoverflow.com/a/15281070/1529139
|
|
// and http://stackoverflow.com/q/40059902/1529139
|
|
bool Utils::signalCtrl(DWORD dwProcessId, DWORD dwCtrlEvent)
|
|
{
|
|
bool success = false;
|
|
DWORD thisConsoleId = GetCurrentProcessId();
|
|
// Leave current console if it exists
|
|
// (otherwise AttachConsole will return ERROR_ACCESS_DENIED)
|
|
bool consoleDetached = (FreeConsole() != FALSE);
|
|
|
|
if (AttachConsole(dwProcessId) != FALSE) {
|
|
// Add a fake Ctrl-C handler for avoid instant kill is this console
|
|
// WARNING: do not revert it or current program will be also killed
|
|
SetConsoleCtrlHandler(nullptr, true);
|
|
success = (GenerateConsoleCtrlEvent(dwCtrlEvent, 0) != FALSE);
|
|
FreeConsole();
|
|
}
|
|
|
|
if (consoleDetached) {
|
|
// Create a new console if previous was deleted by OS
|
|
if (AttachConsole(thisConsoleId) == FALSE) {
|
|
int errorCode = GetLastError();
|
|
if (errorCode == 31) // 31=ERROR_GEN_FAILURE
|
|
{
|
|
AllocConsole();
|
|
}
|
|
}
|
|
}
|
|
return success;
|
|
}
|
|
|
|
#endif
|
|
|
|
void Utils::logException(const std::exception &e)
|
|
{
|
|
qCritical() << e.what();
|
|
try {
|
|
std::rethrow_if_nested(e);
|
|
} catch (const std::exception &nested) {
|
|
logException(nested);
|
|
} catch (...) {}
|
|
}
|
|
|
|
void Utils::logException(const std::exception_ptr &eptr)
|
|
{
|
|
try {
|
|
if (eptr) std::rethrow_exception(eptr);
|
|
} catch (const std::exception &e) {
|
|
logException(e);
|
|
} catch (...) {}
|
|
}
|