/* * Copyright (C) 2012 Tobias Tangemann * Copyright (C) 2012 Felix Geyer * Copyright (C) 2017 KeePassXC Team * * 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 . */ #include "Application.h" #include "MainWindow.h" #include #include #include #include #include #include #include "autotype/AutoType.h" #if defined(Q_OS_UNIX) #include #include #include #endif #if defined(Q_OS_UNIX) && !defined(Q_OS_OSX) class XcbEventFilter : public QAbstractNativeEventFilter { public: bool nativeEventFilter(const QByteArray& eventType, void* message, long* result) override { 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") || eventType == QByteArrayLiteral("windows_dispatcher_MSG")) { int retCode = autoType()->callEventFilter(message); if (retCode == 1) { return true; } } return false; } }; #endif Application::Application(int& argc, char** argv) : QApplication(argc, argv) , m_mainWindow(nullptr) #ifdef Q_OS_UNIX , m_unixSignalNotifier(nullptr) #endif , alreadyRunning(false) , lock(nullptr) { #if defined(Q_OS_UNIX) && !defined(Q_OS_OSX) installNativeEventFilter(new XcbEventFilter()); #elif defined(Q_OS_WIN) installNativeEventFilter(new WinEventFilter()); #endif #if defined(Q_OS_UNIX) registerUnixSignals(); #endif QString userName = qgetenv("USER"); if (userName.isEmpty()) { userName = qgetenv("USERNAME"); } QString identifier = "keepassxc"; if (!userName.isEmpty()) { identifier.append("-"); identifier.append(userName); } QString socketName = identifier + ".socket"; QString lockName = identifier + ".lock"; // 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. lock = new QLockFile(QStandardPaths::writableLocation(QStandardPaths::TempLocation) + "/" + lockName); lock->setStaleLockTime(0); lock->tryLock(); switch (lock->error()) { case QLockFile::NoError: server.setSocketOptions(QLocalServer::UserAccessOption); server.listen(socketName); connect(&server, SIGNAL(newConnection()), this, SIGNAL(anotherInstanceStarted())); break; case QLockFile::LockFailedError: { alreadyRunning = true; // notify the other instance // try several times, in case the other instance is still starting up QLocalSocket client; for (int i = 0; i < 3; i++) { client.connectToServer(socketName); if (client.waitForConnected(150)) { client.abort(); break; } } break; } default: qWarning() << QCoreApplication::translate("Main", "The lock file could not be created. Single-instance mode disabled.") .toUtf8().constData(); } } Application::~Application() { server.close(); if (lock) { lock->unlock(); delete lock; } } QWidget* Application::mainWindow() const { return m_mainWindow; } void Application::setMainWindow(QWidget* mainWindow) { m_mainWindow = mainWindow; } bool Application::event(QEvent* event) { // Handle Apple QFileOpenEvent from finder (double click on .kdbx file) if (event->type() == QEvent::FileOpen) { emit openFile(static_cast(event)->file()); return true; } #ifdef Q_OS_MAC // restore main window when clicking on the docker icon else if ((event->type() == QEvent::ApplicationActivate) && m_mainWindow) { m_mainWindow->ensurePolished(); m_mainWindow->setWindowState(m_mainWindow->windowState() & ~Qt::WindowMinimized); m_mainWindow->show(); m_mainWindow->raise(); m_mainWindow->activateWindow(); } #endif 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; } QVector const handledSignals = { SIGQUIT, SIGINT, SIGTERM, SIGHUP }; for (auto s: handledSignals) { struct sigaction sigAction; sigAction.sa_handler = handleUnixSignal; sigemptyset(&sigAction.sa_mask); sigAction.sa_flags = 0 | SA_RESTART; sigaction(s, &sigAction, nullptr); } 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) { 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; Q_UNUSED(::read(unixSignalSocket[1], &buf, sizeof(buf))); if (nullptr != m_mainWindow) static_cast(m_mainWindow)->appExit(); } #endif bool Application::isAlreadyRunning() const { return alreadyRunning; }