2012-05-10 10:42:17 -04:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2012 Tobias Tangemann
|
2012-07-12 16:33:20 -04:00
|
|
|
* Copyright (C) 2012 Felix Geyer <debfx@fobos.de>
|
2017-06-09 17:40:36 -04:00
|
|
|
* Copyright (C) 2017 KeePassXC Team <team@keepassxc.org>
|
2012-05-10 10:42:17 -04:00
|
|
|
*
|
|
|
|
* 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 or (at your option)
|
|
|
|
* version 3 of the License.
|
|
|
|
*
|
|
|
|
* 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, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2012-05-17 18:35:24 -04:00
|
|
|
#include "Application.h"
|
2020-03-08 22:45:51 -04:00
|
|
|
|
|
|
|
#include "autotype/AutoType.h"
|
2017-07-18 13:17:14 -04:00
|
|
|
#include "core/Config.h"
|
2020-03-08 22:45:51 -04:00
|
|
|
#include "core/Global.h"
|
|
|
|
#include "gui/MainWindow.h"
|
|
|
|
#include "gui/osutils/OSUtils.h"
|
|
|
|
#include "gui/styles/dark/DarkStyle.h"
|
|
|
|
#include "gui/styles/light/LightStyle.h"
|
2012-05-10 10:42:17 -04:00
|
|
|
|
2017-12-05 10:25:08 -05:00
|
|
|
#include <QFileInfo>
|
2013-10-03 09:18:16 -04:00
|
|
|
#include <QFileOpenEvent>
|
2016-07-25 00:41:13 -04:00
|
|
|
#include <QLockFile>
|
2017-12-05 10:25:08 -05:00
|
|
|
#include <QSocketNotifier>
|
2016-07-25 00:41:13 -04:00
|
|
|
#include <QStandardPaths>
|
|
|
|
#include <QtNetwork/QLocalSocket>
|
2012-05-10 10:42:17 -04:00
|
|
|
|
2018-10-30 08:46:12 -04:00
|
|
|
#if defined(Q_OS_WIN) || (defined(Q_OS_UNIX) && !defined(Q_OS_MACOS))
|
|
|
|
#include "core/OSEventFilter.h"
|
|
|
|
#endif
|
|
|
|
|
2017-01-17 18:15:33 -05:00
|
|
|
#if defined(Q_OS_UNIX)
|
|
|
|
#include <signal.h>
|
|
|
|
#include <sys/socket.h>
|
2018-03-31 16:01:30 -04:00
|
|
|
#include <unistd.h>
|
2017-01-17 18:15:33 -05:00
|
|
|
#endif
|
|
|
|
|
2018-03-31 16:01:30 -04:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
constexpr int WaitTimeoutMSec = 150;
|
|
|
|
const char BlockSizeProperty[] = "blockSize";
|
2018-10-30 08:46:12 -04:00
|
|
|
} // namespace
|
2015-07-22 14:30:46 -04:00
|
|
|
|
2012-05-17 18:35:24 -04:00
|
|
|
Application::Application(int& argc, char** argv)
|
2012-05-17 17:59:31 -04:00
|
|
|
: QApplication(argc, argv)
|
2017-01-24 22:24:34 -05:00
|
|
|
#ifdef Q_OS_UNIX
|
|
|
|
, m_unixSignalNotifier(nullptr)
|
|
|
|
#endif
|
2017-09-13 22:54:30 -04:00
|
|
|
, m_alreadyRunning(false)
|
|
|
|
, m_lockFile(nullptr)
|
2018-10-30 08:46:12 -04:00
|
|
|
#if defined(Q_OS_WIN) || (defined(Q_OS_UNIX) && !defined(Q_OS_MACOS))
|
|
|
|
, m_osEventFilter(new OSEventFilter())
|
|
|
|
{
|
|
|
|
installNativeEventFilter(m_osEventFilter.data());
|
|
|
|
#else
|
2012-05-10 10:42:17 -04:00
|
|
|
{
|
2015-07-22 14:30:46 -04:00
|
|
|
#endif
|
2017-01-17 18:15:33 -05:00
|
|
|
#if defined(Q_OS_UNIX)
|
|
|
|
registerUnixSignals();
|
|
|
|
#endif
|
2016-07-25 00:41:13 -04:00
|
|
|
|
|
|
|
QString userName = qgetenv("USER");
|
|
|
|
if (userName.isEmpty()) {
|
|
|
|
userName = qgetenv("USERNAME");
|
|
|
|
}
|
2017-04-18 10:14:23 -04:00
|
|
|
QString identifier = "keepassxc";
|
2016-07-25 00:41:13 -04:00
|
|
|
if (!userName.isEmpty()) {
|
2017-09-13 22:54:30 -04:00
|
|
|
identifier += "-" + userName;
|
|
|
|
}
|
2017-07-18 13:17:14 -04:00
|
|
|
#ifdef QT_DEBUG
|
2018-03-31 16:01:30 -04:00
|
|
|
// In DEBUG mode don't interfere with Release instances
|
|
|
|
identifier += "-DEBUG";
|
2017-07-18 13:17:14 -04:00
|
|
|
#endif
|
2016-07-25 00:41:13 -04:00
|
|
|
QString lockName = identifier + ".lock";
|
2017-12-05 10:25:08 -05:00
|
|
|
m_socketName = identifier + ".socket";
|
2016-07-25 00:41:13 -04:00
|
|
|
|
|
|
|
// According to documentation we should use RuntimeLocation on *nixes, but even Qt doesn't respect
|
|
|
|
// this and creates sockets in TempLocation, so let's be consistent.
|
2017-09-13 22:54:30 -04:00
|
|
|
m_lockFile = new QLockFile(QStandardPaths::writableLocation(QStandardPaths::TempLocation) + "/" + lockName);
|
|
|
|
m_lockFile->setStaleLockTime(0);
|
|
|
|
m_lockFile->tryLock();
|
|
|
|
|
2017-12-05 10:25:08 -05:00
|
|
|
m_lockServer.setSocketOptions(QLocalServer::UserAccessOption);
|
|
|
|
connect(&m_lockServer, SIGNAL(newConnection()), this, SIGNAL(anotherInstanceStarted()));
|
|
|
|
connect(&m_lockServer, SIGNAL(newConnection()), this, SLOT(processIncomingConnection()));
|
|
|
|
|
2017-09-13 22:54:30 -04:00
|
|
|
switch (m_lockFile->error()) {
|
2016-07-25 00:41:13 -04:00
|
|
|
case QLockFile::NoError:
|
2017-09-13 22:54:30 -04:00
|
|
|
// No existing lock was found, start listener
|
2017-12-05 10:25:08 -05:00
|
|
|
m_lockServer.listen(m_socketName);
|
2016-07-25 00:41:13 -04:00
|
|
|
break;
|
|
|
|
case QLockFile::LockFailedError: {
|
2020-04-25 19:31:38 -04:00
|
|
|
if (config()->get(Config::SingleInstance).toBool()) {
|
2017-09-13 22:54:30 -04:00
|
|
|
// Attempt to connect to the existing instance
|
2017-07-18 13:17:14 -04:00
|
|
|
QLocalSocket client;
|
2017-12-05 10:25:08 -05:00
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
client.connectToServer(m_socketName);
|
|
|
|
if (client.waitForConnected(WaitTimeoutMSec)) {
|
2017-09-13 22:54:30 -04:00
|
|
|
// Connection succeeded, this will raise the existing window if minimized
|
2017-07-18 13:17:14 -04:00
|
|
|
client.abort();
|
2017-09-13 22:54:30 -04:00
|
|
|
m_alreadyRunning = true;
|
2017-07-18 13:17:14 -04:00
|
|
|
break;
|
|
|
|
}
|
2016-07-25 00:41:13 -04:00
|
|
|
}
|
2017-09-13 22:54:30 -04:00
|
|
|
|
|
|
|
if (!m_alreadyRunning) {
|
|
|
|
// If we get here then the original instance is likely dead
|
2019-01-30 20:54:35 -05:00
|
|
|
qWarning() << QObject::tr("Existing single-instance lock file is invalid. Launching new instance.")
|
2018-03-31 16:01:30 -04:00
|
|
|
.toUtf8()
|
|
|
|
.constData();
|
2017-09-13 22:54:30 -04:00
|
|
|
|
|
|
|
// forceably reset the lock file
|
|
|
|
m_lockFile->removeStaleLockFile();
|
|
|
|
m_lockFile->tryLock();
|
|
|
|
// start the listen server
|
2017-12-05 10:25:08 -05:00
|
|
|
m_lockServer.listen(m_socketName);
|
2017-09-13 22:54:30 -04:00
|
|
|
}
|
2016-07-25 00:41:13 -04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
2019-03-19 14:48:33 -04:00
|
|
|
qWarning()
|
|
|
|
<< QObject::tr("The lock file could not be created. Single-instance mode disabled.").toUtf8().constData();
|
2016-07-25 00:41:13 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Application::~Application()
|
|
|
|
{
|
2017-09-13 22:54:30 -04:00
|
|
|
m_lockServer.close();
|
|
|
|
if (m_lockFile) {
|
|
|
|
m_lockFile->unlock();
|
|
|
|
delete m_lockFile;
|
2016-07-25 00:41:13 -04:00
|
|
|
}
|
2012-05-10 10:42:17 -04:00
|
|
|
}
|
|
|
|
|
2020-06-04 08:10:43 -04:00
|
|
|
void Application::applyTheme()
|
|
|
|
{
|
2020-07-26 17:02:02 -04:00
|
|
|
auto appTheme = config()->get(Config::GUI_ApplicationTheme).toString();
|
2020-06-04 08:10:43 -04:00
|
|
|
if (appTheme == "auto") {
|
2020-07-26 17:02:02 -04:00
|
|
|
appTheme = osUtils->isDarkMode() ? "dark" : "light";
|
2020-07-28 13:43:32 -04:00
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
if (winUtils()->isHighContrastMode()) {
|
|
|
|
appTheme = "classic";
|
|
|
|
}
|
|
|
|
#endif
|
2020-07-26 17:02:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (appTheme == "light") {
|
2020-06-04 08:10:43 -04:00
|
|
|
setStyle(new LightStyle);
|
2020-07-26 17:02:02 -04:00
|
|
|
// Workaround Qt 5.15+ bug
|
|
|
|
setPalette(style()->standardPalette());
|
2020-06-04 08:10:43 -04:00
|
|
|
} else if (appTheme == "dark") {
|
|
|
|
setStyle(new DarkStyle);
|
2020-07-26 17:02:02 -04:00
|
|
|
// Workaround Qt 5.15+ bug
|
|
|
|
setPalette(style()->standardPalette());
|
2020-06-04 08:10:43 -04:00
|
|
|
m_darkTheme = true;
|
|
|
|
} else {
|
|
|
|
// Classic mode, don't check for dark theme on Windows
|
|
|
|
// because Qt 5.x does not support it
|
|
|
|
#ifndef Q_OS_WIN
|
|
|
|
m_darkTheme = osUtils->isDarkMode();
|
|
|
|
#endif
|
2020-08-01 11:13:28 -04:00
|
|
|
QFile stylesheetFile(":/styles/base/classicstyle.qss");
|
|
|
|
if (stylesheetFile.open(QIODevice::ReadOnly | QIODevice::Text)) {
|
|
|
|
setStyleSheet(stylesheetFile.readAll());
|
|
|
|
stylesheetFile.close();
|
|
|
|
}
|
2020-06-04 08:10:43 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-21 16:11:26 -04:00
|
|
|
bool Application::event(QEvent* event)
|
2012-05-10 10:42:17 -04:00
|
|
|
{
|
2012-05-17 17:59:31 -04:00
|
|
|
// Handle Apple QFileOpenEvent from finder (double click on .kdbx file)
|
|
|
|
if (event->type() == QEvent::FileOpen) {
|
2017-03-10 09:58:42 -05:00
|
|
|
emit openFile(static_cast<QFileOpenEvent*>(event)->file());
|
2012-05-17 17:59:31 -04:00
|
|
|
return true;
|
|
|
|
}
|
2018-10-26 09:19:04 -04:00
|
|
|
#ifdef Q_OS_MACOS
|
2015-10-10 11:00:50 -04:00
|
|
|
// restore main window when clicking on the docker icon
|
2017-12-05 10:25:08 -05:00
|
|
|
else if (event->type() == QEvent::ApplicationActivate) {
|
|
|
|
emit applicationActivated();
|
2015-10-10 11:00:50 -04:00
|
|
|
}
|
|
|
|
#endif
|
2012-05-10 10:42:17 -04:00
|
|
|
|
2012-05-21 16:11:26 -04:00
|
|
|
return QApplication::event(event);
|
2012-05-17 17:59:31 -04:00
|
|
|
}
|
2017-01-17 18:15:33 -05:00
|
|
|
|
|
|
|
#if defined(Q_OS_UNIX)
|
|
|
|
int Application::unixSignalSocket[2];
|
|
|
|
|
|
|
|
void Application::registerUnixSignals()
|
|
|
|
{
|
|
|
|
int result = ::socketpair(AF_UNIX, SOCK_STREAM, 0, unixSignalSocket);
|
|
|
|
Q_ASSERT(0 == result);
|
|
|
|
if (0 != result) {
|
|
|
|
// do not register handles when socket creation failed, otherwise
|
|
|
|
// application will be unresponsive to signals such as SIGINT or SIGTERM
|
|
|
|
return;
|
|
|
|
}
|
2018-03-31 16:01:30 -04:00
|
|
|
|
|
|
|
QVector<int> const handledSignals = {SIGQUIT, SIGINT, SIGTERM, SIGHUP};
|
|
|
|
for (auto s : handledSignals) {
|
2017-01-17 18:15:33 -05:00
|
|
|
struct sigaction sigAction;
|
2018-03-31 16:01:30 -04:00
|
|
|
|
2017-01-17 18:15:33 -05:00
|
|
|
sigAction.sa_handler = handleUnixSignal;
|
|
|
|
sigemptyset(&sigAction.sa_mask);
|
|
|
|
sigAction.sa_flags = 0 | SA_RESTART;
|
|
|
|
sigaction(s, &sigAction, nullptr);
|
|
|
|
}
|
2018-03-31 16:01:30 -04:00
|
|
|
|
2017-01-17 18:15:33 -05:00
|
|
|
m_unixSignalNotifier = new QSocketNotifier(unixSignalSocket[1], QSocketNotifier::Read, this);
|
|
|
|
connect(m_unixSignalNotifier, SIGNAL(activated(int)), this, SLOT(quitBySignal()));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Application::handleUnixSignal(int sig)
|
|
|
|
{
|
|
|
|
switch (sig) {
|
2018-03-31 16:01:30 -04:00
|
|
|
case SIGQUIT:
|
|
|
|
case SIGINT:
|
|
|
|
case SIGTERM: {
|
|
|
|
char buf = 0;
|
|
|
|
Q_UNUSED(::write(unixSignalSocket[0], &buf, sizeof(buf)));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
case SIGHUP:
|
|
|
|
return;
|
2017-01-17 18:15:33 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Application::quitBySignal()
|
|
|
|
{
|
2017-01-17 18:39:36 -05:00
|
|
|
m_unixSignalNotifier->setEnabled(false);
|
2017-01-17 18:15:33 -05:00
|
|
|
char buf;
|
2017-03-10 08:28:26 -05:00
|
|
|
Q_UNUSED(::read(unixSignalSocket[1], &buf, sizeof(buf)));
|
2017-12-05 10:25:08 -05:00
|
|
|
emit quitSignalReceived();
|
2017-01-17 18:15:33 -05:00
|
|
|
}
|
|
|
|
#endif
|
2016-07-25 00:41:13 -04:00
|
|
|
|
2017-12-05 10:25:08 -05:00
|
|
|
void Application::processIncomingConnection()
|
|
|
|
{
|
|
|
|
if (m_lockServer.hasPendingConnections()) {
|
|
|
|
QLocalSocket* socket = m_lockServer.nextPendingConnection();
|
|
|
|
socket->setProperty(BlockSizeProperty, 0);
|
|
|
|
connect(socket, SIGNAL(readyRead()), this, SLOT(socketReadyRead()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Application::socketReadyRead()
|
|
|
|
{
|
|
|
|
QLocalSocket* socket = qobject_cast<QLocalSocket*>(sender());
|
|
|
|
if (!socket) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QDataStream in(socket);
|
|
|
|
in.setVersion(QDataStream::Qt_5_0);
|
|
|
|
|
|
|
|
int blockSize = socket->property(BlockSizeProperty).toInt();
|
|
|
|
if (blockSize == 0) {
|
|
|
|
// Relies on the fact that QDataStream format streams a quint32 into sizeof(quint32) bytes
|
|
|
|
if (socket->bytesAvailable() < qint64(sizeof(quint32))) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
in >> blockSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (socket->bytesAvailable() < blockSize || in.atEnd()) {
|
|
|
|
socket->setProperty(BlockSizeProperty, blockSize);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList fileNames;
|
|
|
|
in >> fileNames;
|
2018-03-31 16:01:30 -04:00
|
|
|
for (const QString& fileName : asConst(fileNames)) {
|
2017-12-05 10:25:08 -05:00
|
|
|
const QFileInfo fInfo(fileName);
|
|
|
|
if (fInfo.isFile() && fInfo.suffix().toLower() == "kdbx") {
|
|
|
|
emit openFile(fileName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
socket->deleteLater();
|
|
|
|
}
|
|
|
|
|
2016-07-25 00:41:13 -04:00
|
|
|
bool Application::isAlreadyRunning() const
|
|
|
|
{
|
2017-07-18 13:17:14 -04:00
|
|
|
#ifdef QT_DEBUG
|
|
|
|
// In DEBUG mode we can run unlimited instances
|
|
|
|
return false;
|
|
|
|
#endif
|
2020-04-25 19:31:38 -04:00
|
|
|
return config()->get(Config::SingleInstance).toBool() && m_alreadyRunning;
|
2016-07-25 00:41:13 -04:00
|
|
|
}
|
|
|
|
|
2017-12-05 10:25:08 -05:00
|
|
|
bool Application::sendFileNamesToRunningInstance(const QStringList& fileNames)
|
|
|
|
{
|
|
|
|
QLocalSocket client;
|
|
|
|
client.connectToServer(m_socketName);
|
|
|
|
const bool connected = client.waitForConnected(WaitTimeoutMSec);
|
|
|
|
if (!connected) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QByteArray data;
|
|
|
|
QDataStream out(&data, QIODevice::WriteOnly);
|
|
|
|
out.setVersion(QDataStream::Qt_5_0);
|
2018-03-31 16:01:30 -04:00
|
|
|
out << quint32(0) << fileNames;
|
2017-12-05 10:25:08 -05:00
|
|
|
out.device()->seek(0);
|
|
|
|
out << quint32(data.size() - sizeof(quint32));
|
|
|
|
|
|
|
|
const bool writeOk = client.write(data) != -1 && client.waitForBytesWritten(WaitTimeoutMSec);
|
|
|
|
client.disconnectFromServer();
|
|
|
|
const bool disconnected = client.waitForDisconnected(WaitTimeoutMSec);
|
|
|
|
return writeOk && disconnected;
|
|
|
|
}
|
2020-03-08 22:45:51 -04:00
|
|
|
|
|
|
|
bool Application::isDarkTheme() const
|
|
|
|
{
|
|
|
|
return m_darkTheme;
|
|
|
|
}
|
2020-06-28 10:01:24 -04:00
|
|
|
|
|
|
|
void Application::restart()
|
|
|
|
{
|
|
|
|
// Disable single instance
|
|
|
|
m_lockServer.close();
|
|
|
|
if (m_lockFile) {
|
|
|
|
m_lockFile->unlock();
|
|
|
|
delete m_lockFile;
|
2020-06-29 08:23:51 -04:00
|
|
|
m_lockFile = nullptr;
|
2020-06-28 10:01:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
exit(RESTART_EXITCODE);
|
|
|
|
}
|