253 lines
		
	
	
	
		
			6.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			253 lines
		
	
	
	
		
			6.7 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 "platform/platform_launcher.h"
 | 
						|
#include "platform/platform_specific.h"
 | 
						|
#include "core/crash_reports.h"
 | 
						|
#include "core/main_queue_processor.h"
 | 
						|
#include "core/update_checker.h"
 | 
						|
#include "base/concurrent_timer.h"
 | 
						|
#include "application.h"
 | 
						|
 | 
						|
namespace Core {
 | 
						|
 | 
						|
std::unique_ptr<Launcher> Launcher::Create(int argc, char *argv[]) {
 | 
						|
	return std::make_unique<Platform::Launcher>(argc, argv);
 | 
						|
}
 | 
						|
 | 
						|
Launcher::Launcher(
 | 
						|
	int argc,
 | 
						|
	char *argv[],
 | 
						|
	const QString &deviceModel,
 | 
						|
	const QString &systemVersion)
 | 
						|
: _argc(argc)
 | 
						|
, _argv(argv)
 | 
						|
, _deviceModel(deviceModel)
 | 
						|
, _systemVersion(systemVersion) {
 | 
						|
}
 | 
						|
 | 
						|
void Launcher::init() {
 | 
						|
	_arguments = readArguments(_argc, _argv);
 | 
						|
 | 
						|
	prepareSettings();
 | 
						|
 | 
						|
	QCoreApplication::setApplicationName(qsl("TelegramDesktop"));
 | 
						|
 | 
						|
#ifndef OS_MAC_OLD
 | 
						|
	QCoreApplication::setAttribute(Qt::AA_DisableHighDpiScaling, true);
 | 
						|
#endif // OS_MAC_OLD
 | 
						|
 | 
						|
	initHook();
 | 
						|
}
 | 
						|
 | 
						|
int Launcher::exec() {
 | 
						|
	init();
 | 
						|
 | 
						|
	if (cLaunchMode() == LaunchModeFixPrevious) {
 | 
						|
		return psFixPrevious();
 | 
						|
	} else if (cLaunchMode() == LaunchModeCleanup) {
 | 
						|
		return psCleanup();
 | 
						|
	}
 | 
						|
 | 
						|
	// both are finished in Application::closeApplication
 | 
						|
	Logs::start(this); // must be started before Platform is started
 | 
						|
	Platform::start(); // must be started before QApplication is created
 | 
						|
 | 
						|
	auto result = executeApplication();
 | 
						|
 | 
						|
	DEBUG_LOG(("Telegram finished, result: %1").arg(result));
 | 
						|
 | 
						|
	if (!UpdaterDisabled() && cRestartingUpdate()) {
 | 
						|
		DEBUG_LOG(("Application Info: executing updater to install update..."));
 | 
						|
		if (!launchUpdater(UpdaterLaunch::PerformUpdate)) {
 | 
						|
			psDeleteDir(cWorkingDir() + qsl("tupdates/temp"));
 | 
						|
		}
 | 
						|
	} else if (cRestarting()) {
 | 
						|
		DEBUG_LOG(("Application Info: executing Telegram, because of restart..."));
 | 
						|
		launchUpdater(UpdaterLaunch::JustRelaunch);
 | 
						|
	}
 | 
						|
 | 
						|
	CrashReports::Finish();
 | 
						|
	Platform::finish();
 | 
						|
	Logs::finish();
 | 
						|
 | 
						|
	return result;
 | 
						|
}
 | 
						|
 | 
						|
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(fromUtf8Safe(argv[i]));
 | 
						|
	}
 | 
						|
	return result;
 | 
						|
}
 | 
						|
 | 
						|
QString Launcher::argumentsString() const {
 | 
						|
	return _arguments.join(' ');
 | 
						|
}
 | 
						|
 | 
						|
bool Launcher::customWorkingDir() const {
 | 
						|
	return _customWorkingDir;
 | 
						|
}
 | 
						|
 | 
						|
void Launcher::prepareSettings() {
 | 
						|
#ifdef Q_OS_MAC
 | 
						|
#ifndef OS_MAC_OLD
 | 
						|
	if (QSysInfo::macVersion() >= QSysInfo::MV_10_11) {
 | 
						|
		gIsElCapitan = true;
 | 
						|
	}
 | 
						|
#else // OS_MAC_OLD
 | 
						|
	if (QSysInfo::macVersion() < QSysInfo::MV_10_7) {
 | 
						|
		gIsSnowLeopard = true;
 | 
						|
	}
 | 
						|
#endif // OS_MAC_OLD
 | 
						|
#endif // Q_OS_MAC
 | 
						|
 | 
						|
	switch (cPlatform()) {
 | 
						|
	case dbipWindows:
 | 
						|
#ifndef OS_WIN_STORE
 | 
						|
		gPlatformString = qsl("Windows");
 | 
						|
#else // OS_WIN_STORE
 | 
						|
		gPlatformString = qsl("WinStore");
 | 
						|
#endif // OS_WIN_STORE
 | 
						|
	break;
 | 
						|
	case dbipMac:
 | 
						|
#ifndef OS_MAC_STORE
 | 
						|
		gPlatformString = qsl("MacOS");
 | 
						|
#else // OS_MAC_STORE
 | 
						|
		gPlatformString = qsl("MacAppStore");
 | 
						|
#endif // OS_MAC_STORE
 | 
						|
	break;
 | 
						|
	case dbipMacOld:
 | 
						|
		gPlatformString = qsl("MacOSold");
 | 
						|
	break;
 | 
						|
	case dbipLinux64:
 | 
						|
		gPlatformString = qsl("Linux64bit");
 | 
						|
	break;
 | 
						|
	case dbipLinux32:
 | 
						|
		gPlatformString = qsl("Linux32bit");
 | 
						|
	break;
 | 
						|
	}
 | 
						|
 | 
						|
	auto path = Platform::CurrentExecutablePath(_argc, _argv);
 | 
						|
	LOG(("Executable path before check: %1").arg(path));
 | 
						|
	if (!path.isEmpty()) {
 | 
						|
		auto info = QFileInfo(path);
 | 
						|
		if (info.isSymLink()) {
 | 
						|
			info = info.symLinkTarget();
 | 
						|
		}
 | 
						|
		if (info.exists()) {
 | 
						|
			const auto dir = info.absoluteDir().absolutePath();
 | 
						|
			gExeDir = (dir.endsWith('/') ? dir : (dir + '/'));
 | 
						|
			gExeName = info.fileName();
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if (cExeName().isEmpty()) {
 | 
						|
		LOG(("WARNING: Could not compute executable path, some features will be disabled."));
 | 
						|
	}
 | 
						|
 | 
						|
	processArguments();
 | 
						|
}
 | 
						|
 | 
						|
QString Launcher::deviceModel() const {
 | 
						|
	return _deviceModel;
 | 
						|
}
 | 
						|
 | 
						|
QString Launcher::systemVersion() const {
 | 
						|
	return _systemVersion;
 | 
						|
}
 | 
						|
 | 
						|
void Launcher::processArguments() {
 | 
						|
		enum class KeyFormat {
 | 
						|
		NoValues,
 | 
						|
		OneValue,
 | 
						|
		AllLeftValues,
 | 
						|
	};
 | 
						|
	auto parseMap = std::map<QByteArray, KeyFormat> {
 | 
						|
		{ "-testmode"       , KeyFormat::NoValues },
 | 
						|
		{ "-debug"          , KeyFormat::NoValues },
 | 
						|
		{ "-many"           , KeyFormat::NoValues },
 | 
						|
		{ "-key"            , KeyFormat::OneValue },
 | 
						|
		{ "-autostart"      , KeyFormat::NoValues },
 | 
						|
		{ "-fixprevious"    , KeyFormat::NoValues },
 | 
						|
		{ "-cleanup"        , KeyFormat::NoValues },
 | 
						|
		{ "-noupdate"       , KeyFormat::NoValues },
 | 
						|
		{ "-externalupdater", KeyFormat::NoValues },
 | 
						|
		{ "-tosettings"     , KeyFormat::NoValues },
 | 
						|
		{ "-startintray"    , KeyFormat::NoValues },
 | 
						|
		{ "-sendpath"       , KeyFormat::AllLeftValues },
 | 
						|
		{ "-workdir"        , KeyFormat::OneValue },
 | 
						|
		{ "--"              , KeyFormat::OneValue },
 | 
						|
	};
 | 
						|
	auto parseResult = QMap<QByteArray, QStringList>();
 | 
						|
	auto parsingKey = QByteArray();
 | 
						|
	auto parsingFormat = KeyFormat::NoValues;
 | 
						|
	for (const auto &argument : _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;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (parseResult.contains("-externalupdater")) {
 | 
						|
		SetUpdaterDisabledAtStartup();
 | 
						|
	}
 | 
						|
	gTestMode = parseResult.contains("-testmode");
 | 
						|
	Logs::SetDebugEnabled(parseResult.contains("-debug"));
 | 
						|
	gManyInstance = parseResult.contains("-many");
 | 
						|
	gKeyFile = parseResult.value("-key", {}).join(QString()).toLower();
 | 
						|
	gKeyFile = gKeyFile.replace(QRegularExpression("[^a-z0-9\\-_]"), {});
 | 
						|
	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");
 | 
						|
	gSendPaths = parseResult.value("-sendpath", {});
 | 
						|
	gWorkingDir = parseResult.value("-workdir", {}).join(QString());
 | 
						|
	if (!gWorkingDir.isEmpty()) {
 | 
						|
		if (QDir().exists(gWorkingDir)) {
 | 
						|
			_customWorkingDir = true;
 | 
						|
		} else {
 | 
						|
			gWorkingDir = QString();
 | 
						|
		}
 | 
						|
	}
 | 
						|
	gStartUrl = parseResult.value("--", {}).join(QString());
 | 
						|
}
 | 
						|
 | 
						|
int Launcher::executeApplication() {
 | 
						|
	MainQueueProcessor processor;
 | 
						|
	base::ConcurrentTimerEnvironment environment;
 | 
						|
 | 
						|
	Application app(this, _argc, _argv);
 | 
						|
	return app.exec();
 | 
						|
}
 | 
						|
 | 
						|
} // namespace Core
 |