624 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			624 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| This file is part of Telegram Desktop,
 | |
| the official desktop application for the Telegram messaging service.
 | |
| 
 | |
| For license and copyright information please follow this link:
 | |
| https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | |
| */
 | |
| #include "core/launcher.h"
 | |
| 
 | |
| #include "kotato/kotato_radius.h"
 | |
| #include "kotato/kotato_settings.h"
 | |
| #include "kotato/kotato_version.h"
 | |
| #include "platform/platform_launcher.h"
 | |
| #include "platform/platform_specific.h"
 | |
| #include "base/options.h"
 | |
| #include "base/platform/base_platform_info.h"
 | |
| #include "base/platform/base_platform_file_utilities.h"
 | |
| #include "ui/main_queue_processor.h"
 | |
| #include "core/crash_reports.h"
 | |
| #include "core/update_checker.h"
 | |
| #include "core/sandbox.h"
 | |
| #include "base/concurrent_timer.h"
 | |
| #include "base/options.h"
 | |
| 
 | |
| #include <QtCore/QLoggingCategory>
 | |
| #include <QtCore/QStandardPaths>
 | |
| 
 | |
| namespace Core {
 | |
| namespace {
 | |
| 
 | |
| constexpr auto kApiIdVarName = "KTGDESKTOP_API_ID"_cs;
 | |
| constexpr auto kApiHashVarName = "KTGDESKTOP_API_HASH"_cs;
 | |
| 
 | |
| uint64 InstallationTag = 0;
 | |
| 
 | |
| base::options::toggle OptionFreeType({
 | |
| 	.id = kOptionFreeType,
 | |
| 	.name = "FreeType font engine",
 | |
| 	.description = "Use the font engine from Linux instead of the system one.",
 | |
| 	.scope = base::options::windows | base::options::macos,
 | |
| 	.restartRequired = true,
 | |
| });
 | |
| 
 | |
| class FilteredCommandLineArguments {
 | |
| public:
 | |
| 	FilteredCommandLineArguments(int argc, char **argv);
 | |
| 
 | |
| 	int &count();
 | |
| 	char **values();
 | |
| 
 | |
| private:
 | |
| 	static constexpr auto kForwardArgumentCount = 1;
 | |
| 
 | |
| 	int _count = 0;
 | |
| 	std::vector<QByteArray> _owned;
 | |
| 	std::vector<char*> _arguments;
 | |
| 
 | |
| 	void pushArgument(const char *text);
 | |
| 
 | |
| };
 | |
| 
 | |
| FilteredCommandLineArguments::FilteredCommandLineArguments(
 | |
| 	int argc,
 | |
| 	char **argv) {
 | |
| 	// For now just pass only the first argument, the executable path.
 | |
| 	for (auto i = 0; i != kForwardArgumentCount; ++i) {
 | |
| 		pushArgument(argv[i]);
 | |
| 	}
 | |
| 
 | |
| #if defined Q_OS_WIN || defined Q_OS_MAC
 | |
| 	if (OptionFreeType.value()) {
 | |
| 		pushArgument("-platform");
 | |
| #ifdef Q_OS_WIN
 | |
| 		pushArgument("windows:fontengine=freetype");
 | |
| #else // Q_OS_WIN
 | |
| 		pushArgument("cocoa:fontengine=freetype");
 | |
| #endif // !Q_OS_WIN
 | |
| 	}
 | |
| #endif // Q_OS_WIN || Q_OS_MAC
 | |
| 
 | |
| 	pushArgument(nullptr);
 | |
| }
 | |
| 
 | |
| int &FilteredCommandLineArguments::count() {
 | |
| 	_count = _arguments.size() - 1;
 | |
| 	return _count;
 | |
| }
 | |
| 
 | |
| char **FilteredCommandLineArguments::values() {
 | |
| 	return _arguments.data();
 | |
| }
 | |
| 
 | |
| void FilteredCommandLineArguments::pushArgument(const char *text) {
 | |
| 	_owned.emplace_back(text);
 | |
| 	_arguments.push_back(_owned.back().data());
 | |
| }
 | |
| 
 | |
| QString DebugModeSettingPath() {
 | |
| 	return cWorkingDir() + u"tdata/withdebug"_q;
 | |
| }
 | |
| 
 | |
| void WriteDebugModeSetting() {
 | |
| 	auto file = QFile(DebugModeSettingPath());
 | |
| 	if (file.open(QIODevice::WriteOnly)) {
 | |
| 		file.write(Logs::DebugEnabled() ? "1" : "0");
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ComputeDebugMode() {
 | |
| 	Logs::SetDebugEnabled(cAlphaVersion() != 0);
 | |
| 	const auto debugModeSettingPath = DebugModeSettingPath();
 | |
| 	auto file = QFile(debugModeSettingPath);
 | |
| 	if (file.exists() && file.open(QIODevice::ReadOnly)) {
 | |
| 		Logs::SetDebugEnabled(file.read(1) != "0");
 | |
| 	}
 | |
| 	if (cDebugMode()) {
 | |
| 		Logs::SetDebugEnabled(true);
 | |
| 	}
 | |
| 	if (Logs::DebugEnabled()) {
 | |
| 		QLoggingCategory::setFilterRules("qt.qpa.gl.debug=true");
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ComputeExternalUpdater() {
 | |
| 	auto locations = QStandardPaths::standardLocations(
 | |
| 		QStandardPaths::AppDataLocation);
 | |
| 	if (locations.isEmpty()) {
 | |
| 		locations << QString();
 | |
| 	}
 | |
| 	locations[0] = QDir::cleanPath(cWorkingDir());
 | |
| 	locations << QDir::cleanPath(cExeDir());
 | |
| 	for (const auto &location : locations) {
 | |
| 		const auto dir = location + u"/externalupdater.d"_q;
 | |
| 		for (const auto &info : QDir(dir).entryInfoList(QDir::Files)) {
 | |
| 			QFile file(info.absoluteFilePath());
 | |
| 			if (file.open(QIODevice::ReadOnly)) {
 | |
| 				QTextStream fileStream(&file);
 | |
| 				while (!fileStream.atEnd()) {
 | |
| 					const auto path = fileStream.readLine();
 | |
| 					if (path == (cExeDir() + cExeName())) {
 | |
| 						SetUpdaterDisabledAtStartup();
 | |
| 						return;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| QString InstallBetaVersionsSettingPath() {
 | |
| 	return cWorkingDir() + u"tdata/devversion"_q;
 | |
| }
 | |
| 
 | |
| void WriteInstallBetaVersionsSetting() {
 | |
| 	QFile f(InstallBetaVersionsSettingPath());
 | |
| 	if (f.open(QIODevice::WriteOnly)) {
 | |
| 		f.write(cInstallBetaVersion() ? "1" : "0");
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ComputeInstallBetaVersions() {
 | |
| 	const auto installBetaSettingPath = InstallBetaVersionsSettingPath();
 | |
| 	if (cAlphaVersion()) {
 | |
| 		cSetInstallBetaVersion(false);
 | |
| 	} else if (QFile::exists(installBetaSettingPath)) {
 | |
| 		QFile f(installBetaSettingPath);
 | |
| 		if (f.open(QIODevice::ReadOnly)) {
 | |
| 			cSetInstallBetaVersion(f.read(1) != "0");
 | |
| 		}
 | |
| 	} else if (AppKotatoBetaVersion) {
 | |
| 		WriteInstallBetaVersionsSetting();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ComputeInstallationTag() {
 | |
| 	InstallationTag = 0;
 | |
| 	auto file = QFile(cWorkingDir() + u"tdata/usertag"_q);
 | |
| 	if (file.open(QIODevice::ReadOnly)) {
 | |
| 		const auto result = file.read(
 | |
| 			reinterpret_cast<char*>(&InstallationTag),
 | |
| 			sizeof(uint64));
 | |
| 		if (result != sizeof(uint64)) {
 | |
| 			InstallationTag = 0;
 | |
| 		}
 | |
| 		file.close();
 | |
| 	}
 | |
| 	if (!InstallationTag) {
 | |
| 		auto generator = std::mt19937(std::random_device()());
 | |
| 		auto distribution = std::uniform_int_distribution<uint64>();
 | |
| 		do {
 | |
| 			InstallationTag = distribution(generator);
 | |
| 		} while (!InstallationTag);
 | |
| 
 | |
| 		if (file.open(QIODevice::WriteOnly)) {
 | |
| 			file.write(
 | |
| 				reinterpret_cast<char*>(&InstallationTag),
 | |
| 				sizeof(uint64));
 | |
| 			file.close();
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool MoveLegacyAlphaFolder(const QString &folder, const QString &file) {
 | |
| 	const auto was = cExeDir() + folder;
 | |
| 	const auto now = cExeDir() + u"TelegramForcePortable"_q;
 | |
| 	if (QDir(was).exists() && !QDir(now).exists()) {
 | |
| 		const auto oldFile = was + "/tdata/" + file;
 | |
| 		const auto newFile = was + "/tdata/alpha";
 | |
| 		if (QFile::exists(oldFile) && !QFile::exists(newFile)) {
 | |
| 			if (!QFile(oldFile).copy(newFile)) {
 | |
| 				LOG(("FATAL: Could not copy '%1' to '%2'").arg(
 | |
| 					oldFile,
 | |
| 					newFile));
 | |
| 				return false;
 | |
| 			}
 | |
| 		}
 | |
| 		if (!QDir().rename(was, now)) {
 | |
| 			LOG(("FATAL: Could not rename '%1' to '%2'").arg(was, now));
 | |
| 			return false;
 | |
| 		}
 | |
| 	}
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool MoveLegacyAlphaFolder() {
 | |
| 	if (!MoveLegacyAlphaFolder(u"TelegramAlpha_data"_q, u"alpha"_q)
 | |
| 		|| !MoveLegacyAlphaFolder(u"TelegramBeta_data"_q, u"beta"_q)) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool CheckPortableVersionFolder() {
 | |
| 	if (!MoveLegacyAlphaFolder()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	const auto portable = cExeDir() + u"TelegramForcePortable"_q;
 | |
| 	QFile key(portable + u"/tdata/alpha"_q);
 | |
| 	if (cAlphaVersion()) {
 | |
| 		/*
 | |
| 		Assert(*AlphaPrivateKey != 0);
 | |
| 		*/
 | |
| 
 | |
| 		cForceWorkingDir(portable);
 | |
| 		QDir().mkpath(cWorkingDir() + u"tdata"_q);
 | |
| 		/*
 | |
| 		cSetAlphaPrivateKey(QByteArray(AlphaPrivateKey));
 | |
| 		if (!key.open(QIODevice::WriteOnly)) {
 | |
| 			LOG(("FATAL: Could not open '%1' for writing private key!"
 | |
| 				).arg(key.fileName()));
 | |
| 			return false;
 | |
| 		}
 | |
| 		QDataStream dataStream(&key);
 | |
| 		dataStream.setVersion(QDataStream::Qt_5_3);
 | |
| 		dataStream << quint64(cRealAlphaVersion()) << cAlphaPrivateKey();
 | |
| 		*/
 | |
| 		return true;
 | |
| 	}
 | |
| 	if (!QDir(portable).exists()) {
 | |
| 		return true;
 | |
| 	}
 | |
| 	cForceWorkingDir(portable);
 | |
| 	if (!key.exists()) {
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	if (!key.open(QIODevice::ReadOnly)) {
 | |
| 		LOG(("FATAL: could not open '%1' for reading private key. "
 | |
| 			"Delete it or reinstall private alpha version."
 | |
| 			).arg(key.fileName()));
 | |
| 		return false;
 | |
| 	}
 | |
| 	QDataStream dataStream(&key);
 | |
| 	dataStream.setVersion(QDataStream::Qt_5_3);
 | |
| 
 | |
| 	quint64 v;
 | |
| 	QByteArray k;
 | |
| 	dataStream >> v >> k;
 | |
| 	if (dataStream.status() != QDataStream::Ok || k.isEmpty()) {
 | |
| 		LOG(("FATAL: '%1' is corrupted. "
 | |
| 			"Delete it or reinstall private alpha version."
 | |
| 			).arg(key.fileName()));
 | |
| 		return false;
 | |
| 	}
 | |
| 	cSetAlphaVersion(AppVersion * 1000ULL);
 | |
| 	cSetAlphaPrivateKey(k);
 | |
| 	cSetRealAlphaVersion(v);
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| base::options::toggle OptionFractionalScalingEnabled({
 | |
| 	.id = kOptionFractionalScalingEnabled,
 | |
| 	.name = "Enable precise High DPI scaling",
 | |
| 	.description = "Follow system interface scale settings exactly.",
 | |
| 	.scope = base::options::windows | base::options::linux,
 | |
| 	.restartRequired = true,
 | |
| });
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| const char kOptionFractionalScalingEnabled[] = "fractional-scaling-enabled";
 | |
| const char kOptionFreeType[] = "freetype";
 | |
| 
 | |
| Launcher *Launcher::InstanceSetter::Instance = nullptr;
 | |
| 
 | |
| std::unique_ptr<Launcher> Launcher::Create(int argc, char *argv[]) {
 | |
| 	return std::make_unique<Platform::Launcher>(argc, argv);
 | |
| }
 | |
| 
 | |
| Launcher::Launcher(int argc, char *argv[])
 | |
| : _argc(argc)
 | |
| , _argv(argv)
 | |
| , _arguments(readArguments(_argc, _argv))
 | |
| , _baseIntegration(_argc, _argv)
 | |
| , _initialWorkingDir(QDir::currentPath() + '/') {
 | |
| 	crl::toggle_fp_exceptions(true);
 | |
| 
 | |
| 	base::Integration::Set(&_baseIntegration);
 | |
| }
 | |
| 
 | |
| Launcher::~Launcher() {
 | |
| 	InstanceSetter::Instance = nullptr;
 | |
| }
 | |
| 
 | |
| void Launcher::init() {
 | |
| 	prepareSettings();
 | |
| 	initQtMessageLogging();
 | |
| 
 | |
| 	QApplication::setApplicationName(u"KotatogramDesktop"_q);
 | |
| 
 | |
| #if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
 | |
| 	// fallback session management is useless for tdesktop since it doesn't have
 | |
| 	// any "are you sure you want to close this window?" dialogs
 | |
| 	// but it produces bugs like https://github.com/telegramdesktop/tdesktop/issues/5022
 | |
| 	// and https://github.com/telegramdesktop/tdesktop/issues/7549
 | |
| 	// and https://github.com/telegramdesktop/tdesktop/issues/948
 | |
| 	// more info: https://doc.qt.io/qt-5/qguiapplication.html#isFallbackSessionManagementEnabled
 | |
| 	QApplication::setFallbackSessionManagementEnabled(false);
 | |
| #endif // Qt < 6.0.0
 | |
| 
 | |
| 	initHook();
 | |
| }
 | |
| 
 | |
| void Launcher::initHighDpi() {
 | |
| #if QT_VERSION < QT_VERSION_CHECK(6, 2, 0)
 | |
| 	qputenv("QT_DPI_ADJUSTMENT_POLICY", "AdjustDpi");
 | |
| #endif // Qt < 6.2.0
 | |
| 
 | |
| #if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
 | |
| 	QApplication::setAttribute(Qt::AA_EnableHighDpiScaling, true);
 | |
| #endif // Qt < 6.0.0
 | |
| 
 | |
| 	if (OptionFractionalScalingEnabled.value()) {
 | |
| 		QApplication::setHighDpiScaleFactorRoundingPolicy(
 | |
| 			Qt::HighDpiScaleFactorRoundingPolicy::PassThrough);
 | |
| 	} else {
 | |
| 		QApplication::setHighDpiScaleFactorRoundingPolicy(
 | |
| 			Qt::HighDpiScaleFactorRoundingPolicy::RoundPreferFloor);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int Launcher::exec() {
 | |
| 	// This should be called before init to load default
 | |
| 	// values and set some options that are not stored in JSON.
 | |
| 	Kotato::JsonSettings::Start();
 | |
| 	init();
 | |
| 
 | |
| 	if (cLaunchMode() == LaunchModeFixPrevious) {
 | |
| 		return psFixPrevious();
 | |
| 	} else if (cLaunchMode() == LaunchModeCleanup) {
 | |
| 		return psCleanup();
 | |
| 	}
 | |
| 
 | |
| 	// Must be started before Platform is started.
 | |
| 	Logs::start();
 | |
| 	base::options::init(cWorkingDir() + "tdata/experimental_options.json");
 | |
| 	Kotato::JsonSettings::Load();
 | |
| 	Kotato::RefreshRadius();
 | |
| 
 | |
| 	// Must be called after options are inited.
 | |
| 	initHighDpi();
 | |
| 
 | |
| 	if (Logs::DebugEnabled()) {
 | |
| 		const auto openalLogPath = QDir::toNativeSeparators(
 | |
| 			cWorkingDir() + u"DebugLogs/last_openal_log.txt"_q);
 | |
| 
 | |
| 		qputenv("ALSOFT_LOGLEVEL", "3");
 | |
| 
 | |
| #ifdef Q_OS_WIN
 | |
| 		_wputenv_s(
 | |
| 			L"ALSOFT_LOGFILE",
 | |
| 			openalLogPath.toStdWString().c_str());
 | |
| #else // Q_OS_WIN
 | |
| 		qputenv(
 | |
| 			"ALSOFT_LOGFILE",
 | |
| 			QFile::encodeName(openalLogPath));
 | |
| #endif // !Q_OS_WIN
 | |
| 	}
 | |
| 
 | |
| 	// Must be started before Sandbox is created.
 | |
| 	Platform::start();
 | |
| 	ThirdParty::start();
 | |
| 
 | |
| 	if (::Kotato::JsonSettings::GetBool("api_use_env")
 | |
| 		&& qEnvironmentVariableIsSet(kApiIdVarName.utf8().constData())
 | |
| 		&& qEnvironmentVariableIsSet(kApiHashVarName.utf8().constData())) {
 | |
| 		::Kotato::JsonSettings::Set("api_id", qgetenv(kApiIdVarName.utf8().constData()).toInt());
 | |
| 		::Kotato::JsonSettings::Set("api_hash", QString::fromLatin1(qgetenv(kApiHashVarName.utf8().constData())));
 | |
| 		::Kotato::JsonSettings::Set("api_start_params", false);
 | |
| 	}
 | |
| 
 | |
| 	auto result = executeApplication();
 | |
| 
 | |
| 	DEBUG_LOG(("Kotatogram finished, result: %1").arg(result));
 | |
| 
 | |
| 	if (!UpdaterDisabled() && cRestartingUpdate()) {
 | |
| 		DEBUG_LOG(("Sandbox Info: executing updater to install update."));
 | |
| 		if (!launchUpdater(UpdaterLaunch::PerformUpdate)) {
 | |
| 			base::Platform::DeleteDirectory(cWorkingDir() + u"tupdates/temp"_q);
 | |
| 		}
 | |
| 	} else if (cRestarting()) {
 | |
| 		DEBUG_LOG(("Sandbox Info: executing Kotatogram because of restart."));
 | |
| 		launchUpdater(UpdaterLaunch::JustRelaunch);
 | |
| 	}
 | |
| 
 | |
| 	CrashReports::Finish();
 | |
| 	ThirdParty::finish();
 | |
| 	Platform::finish();
 | |
| 	Kotato::JsonSettings::Finish();
 | |
| 	Logs::finish();
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| bool Launcher::validateCustomWorkingDir() {
 | |
| 	if (customWorkingDir()) {
 | |
| 		if (_customWorkingDir == cWorkingDir()) {
 | |
| 			_customWorkingDir = QString();
 | |
| 			return false;
 | |
| 		}
 | |
| 		cForceWorkingDir(_customWorkingDir);
 | |
| 		return true;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| void Launcher::workingFolderReady() {
 | |
| 	srand((unsigned int)time(nullptr));
 | |
| 
 | |
| 	ComputeDebugMode();
 | |
| 	ComputeExternalUpdater();
 | |
| 	ComputeInstallBetaVersions();
 | |
| 	ComputeInstallationTag();
 | |
| }
 | |
| 
 | |
| void Launcher::writeDebugModeSetting() {
 | |
| 	WriteDebugModeSetting();
 | |
| }
 | |
| 
 | |
| void Launcher::writeInstallBetaVersionsSetting() {
 | |
| 	WriteInstallBetaVersionsSetting();
 | |
| }
 | |
| 
 | |
| bool Launcher::checkPortableVersionFolder() {
 | |
| 	return CheckPortableVersionFolder();
 | |
| }
 | |
| 
 | |
| QStringList Launcher::readArguments(int argc, char *argv[]) const {
 | |
| 	Expects(argc >= 0);
 | |
| 
 | |
| 	if (const auto native = readArgumentsHook(argc, argv)) {
 | |
| 		return *native;
 | |
| 	}
 | |
| 
 | |
| 	auto result = QStringList();
 | |
| 	result.reserve(argc);
 | |
| 	for (auto i = 0; i != argc; ++i) {
 | |
| 		result.push_back(base::FromUtf8Safe(argv[i]));
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| const QStringList &Launcher::arguments() const {
 | |
| 	return _arguments;
 | |
| }
 | |
| 
 | |
| QString Launcher::initialWorkingDir() const {
 | |
| 	return _initialWorkingDir;
 | |
| }
 | |
| 
 | |
| bool Launcher::customWorkingDir() const {
 | |
| 	return !_customWorkingDir.isEmpty();
 | |
| }
 | |
| 
 | |
| void Launcher::prepareSettings() {
 | |
| 	auto path = base::Platform::CurrentExecutablePath(_argc, _argv);
 | |
| 	LOG(("Executable path before check: %1").arg(path));
 | |
| 	if (cExeName().isEmpty()) {
 | |
| 		LOG(("WARNING: Could not compute executable path, some features will be disabled."));
 | |
| 	}
 | |
| 
 | |
| 	processArguments();
 | |
| }
 | |
| 
 | |
| void Launcher::initQtMessageLogging() {
 | |
| 	static QtMessageHandler OriginalMessageHandler = nullptr;
 | |
| 	OriginalMessageHandler = qInstallMessageHandler([](
 | |
| 			QtMsgType type,
 | |
| 			const QMessageLogContext &context,
 | |
| 			const QString &msg) {
 | |
| 		if (OriginalMessageHandler) {
 | |
| 			OriginalMessageHandler(type, context, msg);
 | |
| 		}
 | |
| 		if (Logs::DebugEnabled() || !Logs::started()) {
 | |
| 			if (!Logs::WritingEntry()) {
 | |
| 				// Sometimes Qt logs something inside our own logging.
 | |
| 				LOG((msg));
 | |
| 			}
 | |
| 		}
 | |
| 	});
 | |
| }
 | |
| 
 | |
| uint64 Launcher::installationTag() const {
 | |
| 	return InstallationTag;
 | |
| }
 | |
| 
 | |
| void Launcher::processArguments() {
 | |
| 	enum class KeyFormat {
 | |
| 		NoValues,
 | |
| 		OneValue,
 | |
| 		AllLeftValues,
 | |
| 	};
 | |
| 	auto parseMap = std::map<QByteArray, KeyFormat> {
 | |
| 		{ "-debug"          , KeyFormat::NoValues },
 | |
| 		{ "-key"            , KeyFormat::OneValue },
 | |
| 		{ "-autostart"      , KeyFormat::NoValues },
 | |
| 		{ "-fixprevious"    , KeyFormat::NoValues },
 | |
| 		{ "-cleanup"        , KeyFormat::NoValues },
 | |
| 		{ "-noupdate"       , KeyFormat::NoValues },
 | |
| 		{ "-tosettings"     , KeyFormat::NoValues },
 | |
| 		{ "-startintray"    , KeyFormat::NoValues },
 | |
| 		{ "-quit"           , KeyFormat::NoValues },
 | |
| 		{ "-sendpath"       , KeyFormat::AllLeftValues },
 | |
| 		{ "-workdir"        , KeyFormat::OneValue },
 | |
| 		{ "--"              , KeyFormat::OneValue },
 | |
| 		{ "-scale"          , KeyFormat::OneValue },
 | |
| 		{ "-no-env-api"     , KeyFormat::NoValues },
 | |
| 		{ "-api-id"         , KeyFormat::OneValue },
 | |
| 		{ "-api-hash"       , KeyFormat::OneValue },
 | |
| 	};
 | |
| 	auto parseResult = QMap<QByteArray, QStringList>();
 | |
| 	auto parsingKey = QByteArray();
 | |
| 	auto parsingFormat = KeyFormat::NoValues;
 | |
| 	for (const auto &argument : std::as_const(_arguments)) {
 | |
| 		switch (parsingFormat) {
 | |
| 		case KeyFormat::OneValue: {
 | |
| 			parseResult[parsingKey] = QStringList(argument.mid(0, 8192));
 | |
| 			parsingFormat = KeyFormat::NoValues;
 | |
| 		} break;
 | |
| 		case KeyFormat::AllLeftValues: {
 | |
| 			parseResult[parsingKey].push_back(argument.mid(0, 8192));
 | |
| 		} break;
 | |
| 		case KeyFormat::NoValues: {
 | |
| 			parsingKey = argument.toLatin1();
 | |
| 			auto it = parseMap.find(parsingKey);
 | |
| 			if (it != parseMap.end()) {
 | |
| 				parsingFormat = it->second;
 | |
| 				parseResult[parsingKey] = QStringList();
 | |
| 			}
 | |
| 		} break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	static const auto RegExp = QRegularExpression("[^a-z0-9\\-_]");
 | |
| 	gDebugMode = parseResult.contains("-debug");
 | |
| 	gKeyFile = parseResult
 | |
| 		.value("-key", {})
 | |
| 		.join(QString())
 | |
| 		.toLower()
 | |
| 		.replace(RegExp, {});
 | |
| 	gLaunchMode = parseResult.contains("-autostart") ? LaunchModeAutoStart
 | |
| 		: parseResult.contains("-fixprevious") ? LaunchModeFixPrevious
 | |
| 		: parseResult.contains("-cleanup") ? LaunchModeCleanup
 | |
| 		: LaunchModeNormal;
 | |
| 	gNoStartUpdate = parseResult.contains("-noupdate");
 | |
| 	gStartToSettings = parseResult.contains("-tosettings");
 | |
| 	gStartInTray = parseResult.contains("-startintray");
 | |
| 	gQuit = parseResult.contains("-quit");
 | |
| 	gSendPaths = parseResult.value("-sendpath", {});
 | |
| 	_customWorkingDir = parseResult.value("-workdir", {}).join(QString());
 | |
| 	if (!_customWorkingDir.isEmpty()) {
 | |
| 		_customWorkingDir = QDir(_customWorkingDir).absolutePath() + '/';
 | |
| 	}
 | |
| 	gStartUrl = parseResult.value("--", {}).join(QString());
 | |
| 
 | |
| 	const auto scaleKey = parseResult.value("-scale", {});
 | |
| 	if (scaleKey.size() > 0) {
 | |
| 		using namespace style;
 | |
| 		const auto value = scaleKey[0].toInt();
 | |
| 		gConfigScale = ((value < kScaleMin) || (value > kScaleMax))
 | |
| 			? kScaleAuto
 | |
| 			: value;
 | |
| 	}
 | |
| 
 | |
| 	::Kotato::JsonSettings::Set("api_use_env", !parseResult.contains("-no-env-api"));
 | |
| 	auto customApiId = parseResult.value("-api-id", {}).join(QString()).toInt();
 | |
| 	auto customApiHash = parseResult.value("-api-hash", {}).join(QString());
 | |
| 	if (customApiId > 0 && !customApiHash.isEmpty()) {
 | |
| 		::Kotato::JsonSettings::Set("api_id", customApiId);
 | |
| 		::Kotato::JsonSettings::Set("api_hash", customApiHash);
 | |
| 		::Kotato::JsonSettings::Set("api_start_params", true);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int Launcher::executeApplication() {
 | |
| 	FilteredCommandLineArguments arguments(_argc, _argv);
 | |
| 	Sandbox sandbox(arguments.count(), arguments.values());
 | |
| 	Ui::MainQueueProcessor processor;
 | |
| 	base::ConcurrentTimerEnvironment environment;
 | |
| 	return sandbox.start();
 | |
| }
 | |
| 
 | |
| } // namespace Core
 |