/* * Bittorrent Client using Qt and libtorrent. * Copyright (C) 2018-2025 Mike Tzou (Chocobo1) * Copyright (C) 2006 Christophe Dumez * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * In addition, as a special exception, the copyright holders give permission to * link this program with the OpenSSL project's "OpenSSL" library (or with * modified versions of it that use the same license as the "OpenSSL" library), * and distribute the linked executables. You must obey the GNU General Public * License in all respects for all of the code used other than "OpenSSL". If you * modify file(s), you may extend this exception to your version of the file(s), * but you are not obligated to do so. If you do not wish to do so, delete this * exception statement from your version. */ #include "foreignapps.h" #if defined(Q_OS_WIN) #include #include #endif #include #include #include #include #if defined(Q_OS_WIN) #include #include #endif #include "base/logger.h" #include "base/preferences.h" #include "base/utils/bytearray.h" #if defined(Q_OS_WIN) #include "base/utils/compare.h" #endif using namespace Utils::ForeignApps; namespace { bool testPythonInstallation(const Path &exePath, PythonInfo &info) { info = {}; QProcess proc; #ifdef Q_OS_UNIX proc.setUnixProcessParameters(QProcess::UnixProcessFlag::CloseFileDescriptors); #endif proc.start(exePath.data(), {u"--version"_s}, QIODevice::ReadOnly); if (proc.waitForFinished() && (proc.exitCode() == QProcess::NormalExit)) { QByteArray procOutput = proc.readAllStandardOutput(); if (procOutput.isEmpty()) procOutput = proc.readAllStandardError(); procOutput = procOutput.simplified(); // Software 'Anaconda' installs its own python interpreter // and `python --version` returns a string like this: // "Python 3.4.3 :: Anaconda 2.3.0 (64-bit)" const QList outputSplit = Utils::ByteArray::splitToViews(procOutput, " "); if (outputSplit.size() <= 1) return false; // User reports: `python --version` -> "Python 3.6.6+" // So trim off unrelated characters const auto versionStr = QString::fromLocal8Bit(outputSplit[1]); const qsizetype idx = versionStr.indexOf(QRegularExpression(u"[^\\.\\d]"_s)); const auto version = PythonInfo::Version::fromString(versionStr.left(idx)); if (!version.isValid()) return false; info = {.executablePath = exePath, .version = version}; LogMsg(QCoreApplication::translate("Utils::ForeignApps", "Found Python executable. Name: \"%1\". Version: \"%2\"") .arg(info.executablePath.toString(), info.version.toString()), Log::INFO); return true; } return false; } #if defined(Q_OS_WIN) enum REG_SEARCH_TYPE { USER, SYSTEM_32BIT, SYSTEM_64BIT }; PathList getRegSubkeys(const HKEY handle) { PathList keys; DWORD cSubKeys = 0; DWORD cMaxSubKeyLen = 0; const LSTATUS result = ::RegQueryInfoKeyW(handle, NULL, NULL, NULL, &cSubKeys, &cMaxSubKeyLen, NULL, NULL, NULL, NULL, NULL, NULL); if (result == ERROR_SUCCESS) { ++cMaxSubKeyLen; // For null character LPWSTR lpName = new WCHAR[cMaxSubKeyLen] {0}; [[maybe_unused]] const auto lpNameGuard = qScopeGuard([&lpName] { delete[] lpName; }); keys.reserve(cSubKeys); for (DWORD i = 0; i < cSubKeys; ++i) { DWORD cName = cMaxSubKeyLen; const LSTATUS res = ::RegEnumKeyExW(handle, i, lpName, &cName, NULL, NULL, NULL, NULL); if (res == ERROR_SUCCESS) keys.append(Path(QString::fromWCharArray(lpName))); } } return keys; } Path getRegValue(const HKEY handle, const QString &name = {}) { const std::wstring nameWStr = name.toStdWString(); DWORD type = 0; DWORD cbData = 0; // Discover the size of the value ::RegQueryValueExW(handle, nameWStr.c_str(), NULL, &type, NULL, &cbData); const DWORD cBuffer = (cbData / sizeof(WCHAR)) + 1; LPWSTR lpData = new WCHAR[cBuffer] {0}; [[maybe_unused]] const auto lpDataGuard = qScopeGuard([&lpData] { delete[] lpData; }); const LSTATUS res = ::RegQueryValueExW(handle, nameWStr.c_str(), NULL, &type, reinterpret_cast(lpData), &cbData); if (res == ERROR_SUCCESS) return Path(QString::fromWCharArray(lpData)); return {}; } PathList pythonSearchReg(const REG_SEARCH_TYPE type) { const HKEY hkRoot = (type == USER) ? HKEY_CURRENT_USER : HKEY_LOCAL_MACHINE; const REGSAM samDesired = KEY_READ | ((type == SYSTEM_64BIT) ? KEY_WOW64_64KEY : KEY_WOW64_32KEY); PathList ret; HKEY hkPythonCore {0}; if (::RegOpenKeyExW(hkRoot, L"SOFTWARE\\Python\\PythonCore", 0, samDesired, &hkPythonCore) == ERROR_SUCCESS) { [[maybe_unused]] const auto hkPythonCoreGuard = qScopeGuard([&hkPythonCore] { ::RegCloseKey(hkPythonCore); }); // start with the largest version PathList versions = getRegSubkeys(hkPythonCore); // ordinary sort won't suffice, it needs to sort ["3.9", "3.10"] correctly const Utils::Compare::NaturalCompare comparator; std::sort(versions.begin(), versions.end(), [&comparator](const Path &left, const Path &right) { return comparator(left.data(), right.data()); }); ret.reserve(versions.size() * 2); while (!versions.empty()) { const std::wstring version = (versions.takeLast() / Path(u"InstallPath"_s)).toString().toStdWString(); HKEY hkInstallPath {0}; if (::RegOpenKeyExW(hkPythonCore, version.c_str(), 0, samDesired, &hkInstallPath) == ERROR_SUCCESS) { [[maybe_unused]] const auto hkInstallPathGuard = qScopeGuard([&hkInstallPath] { ::RegCloseKey(hkInstallPath); }); const Path basePath = getRegValue(hkInstallPath); if (basePath.isEmpty()) continue; if (const Path path = (basePath / Path(u"python3.exe"_s)); path.exists()) ret.append(path); if (const Path path = (basePath / Path(u"python.exe"_s)); path.exists()) ret.append(path); } } } return ret; } PathList searchPythonPaths() { // From registry PathList ret = pythonSearchReg(USER) + pythonSearchReg(SYSTEM_64BIT) + pythonSearchReg(SYSTEM_32BIT); // Fallback: Detect python from default locations const QFileInfoList dirs = QDir(u"C:/"_s).entryInfoList({u"Python*"_s}, QDir::Dirs, (QDir::Name | QDir::Reversed)); for (const QFileInfo &info : dirs) { const Path absPath {info.absolutePath()}; if (const Path path = (absPath / Path(u"python3.exe"_s)); path.exists()) ret.append(path); if (const Path path = (absPath / Path(u"python.exe"_s)); path.exists()) ret.append(path); } return ret; } #endif // Q_OS_WIN } bool Utils::ForeignApps::PythonInfo::isValid() const { return (executablePath.isValid() && version.isValid()); } bool Utils::ForeignApps::PythonInfo::isSupportedVersion() const { return (version >= MINIMUM_SUPPORTED_VERSION); } PythonInfo Utils::ForeignApps::pythonInfo() { static PythonInfo pyInfo; const Path preferredPythonPath = Preferences::instance()->getPythonExecutablePath(); if (pyInfo.isValid() && (preferredPythonPath == pyInfo.executablePath)) return pyInfo; const QString invalidVersionMessage = QCoreApplication::translate("Utils::ForeignApps" , "Python failed to meet minimum version requirement. Path: \"%1\". Found version: \"%2\". Minimum supported version: \"%3\"."); if (!preferredPythonPath.isEmpty()) { if (testPythonInstallation(preferredPythonPath, pyInfo)) { if (pyInfo.isSupportedVersion()) return pyInfo; LogMsg(invalidVersionMessage.arg(pyInfo.executablePath.toString() , pyInfo.version.toString(), PythonInfo::MINIMUM_SUPPORTED_VERSION.toString()), Log::WARNING); } else { LogMsg(QCoreApplication::translate("Utils::ForeignApps", "Failed to find Python executable. Path: \"%1\".") .arg(preferredPythonPath.toString()), Log::WARNING); } } else { // auto detect only when there are no preferred python path if (!pyInfo.isValid()) { // search in `PATH` environment variable const QString exeNames[] = {u"python3"_s, u"python"_s}; for (const QString &exeName : exeNames) { if (testPythonInstallation(Path(exeName), pyInfo)) { if (pyInfo.isSupportedVersion()) return pyInfo; LogMsg(invalidVersionMessage.arg(pyInfo.executablePath.toString() , pyInfo.version.toString(), PythonInfo::MINIMUM_SUPPORTED_VERSION.toString()), Log::INFO); } else { LogMsg(QCoreApplication::translate("Utils::ForeignApps", "Failed to find `%1` executable in PATH environment variable. PATH: \"%2\"") .arg(exeName, qEnvironmentVariable("PATH")), Log::INFO); } } #if defined(Q_OS_WIN) for (const Path &path : asConst(searchPythonPaths())) { if (testPythonInstallation(path, pyInfo)) { if (pyInfo.isSupportedVersion()) return pyInfo; LogMsg(invalidVersionMessage.arg(pyInfo.executablePath.toString() , pyInfo.version.toString(), PythonInfo::MINIMUM_SUPPORTED_VERSION.toString()), Log::INFO); } } #endif LogMsg(QCoreApplication::translate("Utils::ForeignApps", "Failed to find Python executable"), Log::WARNING); } } return pyInfo; }