keepassxc/src/gui/Application.cpp

333 lines
9.8 KiB
C++
Raw Normal View History

2012-05-10 10:42:17 -04:00
/*
* Copyright (C) 2012 Tobias Tangemann
* 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/>.
*/
#include "Application.h"
#include "MainWindow.h"
2017-07-18 13:17:14 -04:00
#include "core/Config.h"
2012-05-10 10:42:17 -04:00
2015-07-22 14:30:46 -04:00
#include <QAbstractNativeEventFilter>
#include <QFileInfo>
#include <QFileOpenEvent>
2016-07-25 00:41:13 -04:00
#include <QLockFile>
#include <QSocketNotifier>
2016-07-25 00:41:13 -04:00
#include <QStandardPaths>
#include <QtNetwork/QLocalSocket>
2012-05-10 10:42:17 -04:00
#include "autotype/AutoType.h"
#include "core/Global.h"
#if defined(Q_OS_UNIX)
#include <signal.h>
#include <sys/socket.h>
2018-03-31 16:01:30 -04:00
#include <unistd.h>
#endif
2018-03-31 16:01:30 -04:00
namespace
{
constexpr int WaitTimeoutMSec = 150;
const char BlockSizeProperty[] = "blockSize";
}
#if defined(Q_OS_UNIX) && !defined(Q_OS_MACOS)
2015-07-22 14:30:46 -04:00
class XcbEventFilter : public QAbstractNativeEventFilter
{
public:
2015-10-15 12:02:31 -04:00
bool nativeEventFilter(const QByteArray& eventType, void* message, long* result) override
2015-07-22 14:30:46 -04:00
{
Q_UNUSED(result)
if (eventType == QByteArrayLiteral("xcb_generic_event_t")) {
int retCode = autoType()->callEventFilter(message);
if (retCode == 1) {
return true;
}
}
return false;
}
};
#elif defined(Q_OS_WIN)
class WinEventFilter : public QAbstractNativeEventFilter
{
public:
bool nativeEventFilter(const QByteArray& eventType, void* message, long* result) override
{
Q_UNUSED(result);
if (eventType == QByteArrayLiteral("windows_generic_MSG")
2018-03-31 16:01:30 -04:00
|| eventType == QByteArrayLiteral("windows_dispatcher_MSG")) {
int retCode = autoType()->callEventFilter(message);
if (retCode == 1) {
return true;
}
}
return false;
}
};
2015-07-22 14:30:46 -04:00
#endif
Application::Application(int& argc, char** argv)
: QApplication(argc, argv)
2017-01-24 22:24:34 -05:00
, m_mainWindow(nullptr)
#ifdef Q_OS_UNIX
, m_unixSignalNotifier(nullptr)
#endif
, m_alreadyRunning(false)
, m_lockFile(nullptr)
2012-05-10 10:42:17 -04:00
{
#if defined(Q_OS_UNIX) && !defined(Q_OS_MACOS)
2015-07-22 14:30:46 -04:00
installNativeEventFilter(new XcbEventFilter());
#elif defined(Q_OS_WIN)
installNativeEventFilter(new WinEventFilter());
2015-07-22 14:30:46 -04:00
#endif
#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()) {
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";
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.
m_lockFile = new QLockFile(QStandardPaths::writableLocation(QStandardPaths::TempLocation) + "/" + lockName);
m_lockFile->setStaleLockTime(0);
m_lockFile->tryLock();
m_lockServer.setSocketOptions(QLocalServer::UserAccessOption);
connect(&m_lockServer, SIGNAL(newConnection()), this, SIGNAL(anotherInstanceStarted()));
connect(&m_lockServer, SIGNAL(newConnection()), this, SLOT(processIncomingConnection()));
switch (m_lockFile->error()) {
2016-07-25 00:41:13 -04:00
case QLockFile::NoError:
// No existing lock was found, start listener
m_lockServer.listen(m_socketName);
2016-07-25 00:41:13 -04:00
break;
case QLockFile::LockFailedError: {
2017-07-18 13:17:14 -04:00
if (config()->get("SingleInstance").toBool()) {
// Attempt to connect to the existing instance
2017-07-18 13:17:14 -04:00
QLocalSocket client;
for (int i = 0; i < 3; ++i) {
client.connectToServer(m_socketName);
if (client.waitForConnected(WaitTimeoutMSec)) {
// Connection succeeded, this will raise the existing window if minimized
2017-07-18 13:17:14 -04:00
client.abort();
m_alreadyRunning = true;
2017-07-18 13:17:14 -04:00
break;
}
2016-07-25 00:41:13 -04:00
}
if (!m_alreadyRunning) {
// If we get here then the original instance is likely dead
2018-03-31 16:01:30 -04:00
qWarning() << QCoreApplication::translate(
"Main", "Existing single-instance lock file is invalid. Launching new instance.")
.toUtf8()
.constData();
// forceably reset the lock file
m_lockFile->removeStaleLockFile();
m_lockFile->tryLock();
// start the listen server
m_lockServer.listen(m_socketName);
}
2016-07-25 00:41:13 -04:00
}
break;
}
default:
qWarning() << QCoreApplication::translate("Main",
2018-03-31 16:01:30 -04:00
"The lock file could not be created. Single-instance mode disabled.")
.toUtf8()
.constData();
2016-07-25 00:41:13 -04:00
}
}
Application::~Application()
{
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
}
QWidget* Application::mainWindow() const
{
return m_mainWindow;
}
void Application::setMainWindow(QWidget* mainWindow)
{
m_mainWindow = mainWindow;
}
2012-05-21 16:11:26 -04:00
bool Application::event(QEvent* event)
2012-05-10 10:42:17 -04:00
{
// Handle Apple QFileOpenEvent from finder (double click on .kdbx file)
if (event->type() == QEvent::FileOpen) {
emit openFile(static_cast<QFileOpenEvent*>(event)->file());
return true;
}
#ifdef Q_OS_MACOS
// restore main window when clicking on the docker icon
else if (event->type() == QEvent::ApplicationActivate) {
emit applicationActivated();
}
#endif
2012-05-10 10:42:17 -04:00
2012-05-21 16:11:26 -04:00
return QApplication::event(event);
}
#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) {
struct sigaction sigAction;
2018-03-31 16:01:30 -04: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
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;
}
}
void Application::quitBySignal()
{
m_unixSignalNotifier->setEnabled(false);
char buf;
2017-03-10 08:28:26 -05:00
Q_UNUSED(::read(unixSignalSocket[1], &buf, sizeof(buf)));
emit quitSignalReceived();
}
#endif
2016-07-25 00:41:13 -04: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)) {
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
return config()->get("SingleInstance").toBool() && m_alreadyRunning;
2016-07-25 00:41:13 -04: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;
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;
}