2019-01-07 01:00:58 +01:00
|
|
|
// The MIT License (MIT)
|
|
|
|
//
|
2020-10-05 21:31:03 +02:00
|
|
|
// Copyright (c) Itay Grudev 2015 - 2020
|
2019-01-07 01:00:58 +01:00
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
// THE SOFTWARE.
|
|
|
|
|
2019-08-26 02:00:47 +02:00
|
|
|
//
|
|
|
|
// W A R N I N G !!!
|
|
|
|
// -----------------
|
|
|
|
//
|
|
|
|
// This is a modified version of SingleApplication,
|
|
|
|
// The original version is at:
|
|
|
|
//
|
|
|
|
// https://github.com/itay-grudev/SingleApplication
|
|
|
|
//
|
|
|
|
//
|
|
|
|
|
2020-06-14 23:54:18 +02:00
|
|
|
#include <cstdlib>
|
2020-02-08 03:40:30 +01:00
|
|
|
#include <limits>
|
2023-03-05 00:08:45 +01:00
|
|
|
#include <memory>
|
2020-02-08 03:40:30 +01:00
|
|
|
|
2023-03-04 23:47:36 +01:00
|
|
|
#include <boost/scope_exit.hpp>
|
|
|
|
|
2019-08-26 02:00:47 +02:00
|
|
|
#include <QtGlobal>
|
|
|
|
#include <QThread>
|
|
|
|
#include <QSharedMemory>
|
2020-02-08 03:40:30 +01:00
|
|
|
#include <QLocalSocket>
|
2019-08-26 02:00:47 +02:00
|
|
|
#include <QByteArray>
|
2019-12-21 18:11:54 +01:00
|
|
|
#include <QElapsedTimer>
|
2020-02-08 03:40:30 +01:00
|
|
|
#include <QtDebug>
|
2023-03-02 23:03:33 +01:00
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 6, 0)
|
|
|
|
# include <QNativeIpcKey>
|
|
|
|
#endif
|
2019-01-07 01:00:58 +01:00
|
|
|
|
2023-03-04 19:31:04 +01:00
|
|
|
#include "singleapplication_t.h"
|
2019-01-07 01:00:58 +01:00
|
|
|
#include "singleapplication_p.h"
|
|
|
|
|
|
|
|
/**
|
2020-10-05 21:31:03 +02:00
|
|
|
* @brief Constructor. Checks and fires up LocalServer or closes the program if another instance already exists
|
2019-01-07 01:00:58 +01:00
|
|
|
* @param argc
|
|
|
|
* @param argv
|
2020-10-05 21:31:03 +02:00
|
|
|
* @param allowSecondary Whether to enable secondary instance support
|
|
|
|
* @param options Optional flags to toggle specific behaviour
|
|
|
|
* @param timeout Maximum time blocking functions are allowed during app load
|
2019-01-07 01:00:58 +01:00
|
|
|
*/
|
2023-03-04 19:31:04 +01:00
|
|
|
SingleApplicationClass::SingleApplicationClass(int &argc, char *argv[], const bool allowSecondary, const Options options, const int timeout)
|
|
|
|
: ApplicationClass(argc, argv),
|
|
|
|
d_ptr(new SingleApplicationPrivateClass(this)) {
|
2019-08-26 02:00:47 +02:00
|
|
|
|
2023-03-04 19:31:04 +01:00
|
|
|
#if defined(SINGLEAPPLICATION)
|
2019-08-26 02:00:47 +02:00
|
|
|
Q_D(SingleApplication);
|
2023-03-04 19:31:04 +01:00
|
|
|
#elif defined(SINGLECOREAPPLICATION)
|
|
|
|
Q_D(SingleCoreApplication);
|
|
|
|
#endif
|
2019-01-07 01:00:58 +01:00
|
|
|
|
2019-08-26 02:00:47 +02:00
|
|
|
// Store the current mode of the program
|
2020-10-05 21:31:03 +02:00
|
|
|
d->options_ = options;
|
2019-01-07 01:00:58 +01:00
|
|
|
|
2019-08-26 02:00:47 +02:00
|
|
|
// Generating an application ID used for identifying the shared memory block and QLocalServer
|
|
|
|
d->genBlockServerName();
|
2019-01-07 01:00:58 +01:00
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
// To mitigate QSharedMemory issues with large amount of processes attempting to attach at the same time
|
2023-03-04 19:31:04 +01:00
|
|
|
SingleApplicationPrivateClass::randomSleep();
|
2020-10-05 21:31:03 +02:00
|
|
|
|
2019-01-07 01:00:58 +01:00
|
|
|
#ifdef Q_OS_UNIX
|
2020-10-05 21:31:03 +02:00
|
|
|
// By explicitly attaching it and then deleting it we make sure that the memory is deleted even after the process has crashed on Unix.
|
2023-03-04 23:47:36 +01:00
|
|
|
{
|
|
|
|
# if QT_VERSION >= QT_VERSION_CHECK(6, 6, 0)
|
|
|
|
std::unique_ptr<QSharedMemory> memory = std::make_unique<QSharedMemory>(QNativeIpcKey(d->blockServerName_));
|
|
|
|
# else
|
|
|
|
std::unique_ptr<QSharedMemory> memory = std::make_unique<QSharedMemory>(d->blockServerName_);
|
|
|
|
# endif
|
|
|
|
if (memory->attach()) {
|
|
|
|
memory->detach();
|
|
|
|
}
|
|
|
|
}
|
2019-01-07 01:00:58 +01:00
|
|
|
#endif
|
2020-10-05 21:31:03 +02:00
|
|
|
|
2019-08-26 02:00:47 +02:00
|
|
|
// Guarantee thread safe behaviour with a shared memory block.
|
2023-03-02 22:57:33 +01:00
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 6, 0)
|
2023-03-04 23:47:36 +01:00
|
|
|
QSharedMemory *memory = new QSharedMemory(QNativeIpcKey(d->blockServerName_), this);
|
2023-03-02 22:57:33 +01:00
|
|
|
#else
|
2023-03-04 23:47:36 +01:00
|
|
|
QSharedMemory *memory = new QSharedMemory(d->blockServerName_, this);
|
2023-03-02 22:57:33 +01:00
|
|
|
#endif
|
2023-03-04 23:47:36 +01:00
|
|
|
d->memory_ = memory;
|
|
|
|
|
|
|
|
bool primary = false;
|
2019-08-26 02:00:47 +02:00
|
|
|
|
|
|
|
// Create a shared memory block
|
2023-03-04 23:47:36 +01:00
|
|
|
if (d->memory_->create(sizeof(SingleApplicationPrivateClass::InstancesInfo))) {
|
|
|
|
primary = true;
|
|
|
|
}
|
|
|
|
else if (d->memory_->error() == QSharedMemory::AlreadyExists) {
|
|
|
|
if (!d->memory_->attach()) {
|
|
|
|
qCritical() << "SingleApplication: Unable to attach to shared memory block:" << d->memory_->error() << d->memory_->errorString();
|
|
|
|
return;
|
2020-10-05 21:31:03 +02:00
|
|
|
}
|
2019-08-26 02:00:47 +02:00
|
|
|
}
|
|
|
|
else {
|
2023-03-04 23:47:36 +01:00
|
|
|
qCritical() << "SingleApplication: Unable to create shared memory block:" << d->memory_->error() << d->memory_->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool locked = false;
|
|
|
|
|
|
|
|
BOOST_SCOPE_EXIT((memory)(&locked)) {
|
|
|
|
if (locked && !memory->unlock()) {
|
|
|
|
qWarning() << "SingleApplication: Unable to unlock shared memory block:" << memory->error() << memory->errorString();
|
|
|
|
return;
|
2019-01-07 01:00:58 +01:00
|
|
|
}
|
2023-03-04 23:47:36 +01:00
|
|
|
}BOOST_SCOPE_EXIT_END
|
|
|
|
|
|
|
|
if (!d->memory_->lock()) {
|
|
|
|
qCritical() << "SingleApplication: Unable to lock shared memory block:" << d->memory_->error() << d->memory_->errorString();
|
|
|
|
return;
|
2019-08-26 02:00:47 +02:00
|
|
|
}
|
2023-03-04 23:47:36 +01:00
|
|
|
locked = true;
|
2019-01-07 01:00:58 +01:00
|
|
|
|
2023-03-04 23:47:36 +01:00
|
|
|
if (primary) {
|
|
|
|
// Initialize the shared memory block
|
|
|
|
d->initializeMemoryBlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
SingleApplicationPrivateClass::InstancesInfo *instance = static_cast<SingleApplicationPrivateClass::InstancesInfo*>(d->memory_->data());
|
2019-12-21 18:11:54 +01:00
|
|
|
QElapsedTimer time;
|
2019-08-26 02:00:47 +02:00
|
|
|
time.start();
|
2019-01-07 01:00:58 +01:00
|
|
|
|
2023-03-04 23:47:36 +01:00
|
|
|
// Make sure the shared memory block is initialized and in a consistent state
|
|
|
|
while (d->blockChecksum() != instance->checksum) {
|
2019-01-07 01:00:58 +01:00
|
|
|
|
2023-03-04 23:47:36 +01:00
|
|
|
// If more than 5 seconds have elapsed, assume the primary instance crashed and assume its position
|
2019-08-26 02:00:47 +02:00
|
|
|
if (time.elapsed() > 5000) {
|
2023-03-04 23:47:36 +01:00
|
|
|
qWarning() << "SingleApplication: Shared memory block has been in an inconsistent state from more than 5 seconds. Assuming primary instance failure.";
|
2019-08-26 02:00:47 +02:00
|
|
|
d->initializeMemoryBlock();
|
|
|
|
}
|
2019-01-07 01:00:58 +01:00
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
// Otherwise wait for a random period and try again.
|
2023-03-04 23:47:36 +01:00
|
|
|
// The random sleep here limits the probability of a collision between two racing apps and allows the app to initialize faster
|
|
|
|
if (locked) {
|
|
|
|
if (d->memory_->unlock()) {
|
|
|
|
locked = false;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
qCritical() << "SingleApplication: Unable to unlock shared memory block for random wait:" << memory->error() << memory->errorString();
|
|
|
|
return;
|
|
|
|
}
|
2020-10-05 21:31:03 +02:00
|
|
|
}
|
2023-03-04 23:47:36 +01:00
|
|
|
|
2023-03-04 19:31:04 +01:00
|
|
|
SingleApplicationPrivateClass::randomSleep();
|
2023-03-04 23:47:36 +01:00
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
if (!d->memory_->lock()) {
|
2023-03-04 23:47:36 +01:00
|
|
|
qCritical() << "SingleApplication: Unable to lock shared memory block after random wait:" << memory->error() << memory->errorString();
|
|
|
|
return;
|
2020-10-05 21:31:03 +02:00
|
|
|
}
|
2023-03-04 23:47:36 +01:00
|
|
|
locked = true;
|
|
|
|
|
2019-08-26 02:00:47 +02:00
|
|
|
}
|
2019-01-07 01:00:58 +01:00
|
|
|
|
2023-03-04 23:47:36 +01:00
|
|
|
if (instance->primary) {
|
|
|
|
// Check if another instance can be started
|
|
|
|
if (allowSecondary) {
|
|
|
|
d->startSecondary();
|
|
|
|
if (d->options_ & Mode::SecondaryNotification) {
|
|
|
|
d->connectToPrimary(timeout, SingleApplicationPrivateClass::SecondaryInstance);
|
|
|
|
}
|
2020-10-05 21:31:03 +02:00
|
|
|
}
|
2023-03-04 23:47:36 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
d->startPrimary();
|
|
|
|
primary = true;
|
2019-08-26 02:00:47 +02:00
|
|
|
}
|
|
|
|
|
2023-03-04 23:47:36 +01:00
|
|
|
if (locked) {
|
|
|
|
if (d->memory_->unlock()) {
|
|
|
|
locked = false;
|
2019-01-07 01:00:58 +01:00
|
|
|
}
|
2023-03-04 23:47:36 +01:00
|
|
|
else {
|
|
|
|
qWarning() << "SingleApplication: Unable to unlock shared memory block:" << memory->error() << memory->errorString();
|
2020-10-05 21:31:03 +02:00
|
|
|
}
|
2019-08-26 02:00:47 +02:00
|
|
|
}
|
2019-01-07 01:00:58 +01:00
|
|
|
|
2023-03-04 23:47:36 +01:00
|
|
|
if (!primary && !allowSecondary) {
|
|
|
|
d->connectToPrimary(timeout, SingleApplicationPrivateClass::NewInstance);
|
2020-10-05 21:31:03 +02:00
|
|
|
}
|
2019-01-07 01:00:58 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2023-03-04 19:31:04 +01:00
|
|
|
SingleApplicationClass::~SingleApplicationClass() {
|
|
|
|
|
|
|
|
#if defined(SINGLEAPPLICATION)
|
2019-08-26 02:00:47 +02:00
|
|
|
Q_D(SingleApplication);
|
2023-03-04 19:31:04 +01:00
|
|
|
#elif defined(SINGLECOREAPPLICATION)
|
|
|
|
Q_D(SingleCoreApplication);
|
|
|
|
#endif
|
|
|
|
|
2019-08-26 02:00:47 +02:00
|
|
|
delete d;
|
2023-03-04 19:31:04 +01:00
|
|
|
|
2019-01-07 01:00:58 +01:00
|
|
|
}
|
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
/**
|
|
|
|
* Checks if the current application instance is primary.
|
|
|
|
* @return Returns true if the instance is primary, false otherwise.
|
|
|
|
*/
|
2023-03-04 19:31:04 +01:00
|
|
|
bool SingleApplicationClass::isPrimary() const {
|
|
|
|
|
|
|
|
#if defined(SINGLEAPPLICATION)
|
2022-02-06 00:07:15 +01:00
|
|
|
Q_D(const SingleApplication);
|
2023-03-04 19:31:04 +01:00
|
|
|
#elif defined(SINGLECOREAPPLICATION)
|
|
|
|
Q_D(const SingleCoreApplication);
|
|
|
|
#endif
|
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
return d->server_ != nullptr;
|
2023-03-04 19:31:04 +01:00
|
|
|
|
2019-01-07 01:00:58 +01:00
|
|
|
}
|
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
/**
|
|
|
|
* Checks if the current application instance is secondary.
|
|
|
|
* @return Returns true if the instance is secondary, false otherwise.
|
|
|
|
*/
|
2023-03-04 19:31:04 +01:00
|
|
|
bool SingleApplicationClass::isSecondary() const {
|
|
|
|
|
|
|
|
#if defined(SINGLEAPPLICATION)
|
2022-02-06 00:07:15 +01:00
|
|
|
Q_D(const SingleApplication);
|
2023-03-04 19:31:04 +01:00
|
|
|
#elif defined(SINGLECOREAPPLICATION)
|
|
|
|
Q_D(const SingleCoreApplication);
|
|
|
|
#endif
|
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
return d->server_ == nullptr;
|
2023-03-04 19:31:04 +01:00
|
|
|
|
2019-01-07 01:00:58 +01:00
|
|
|
}
|
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
/**
|
|
|
|
* Allows you to identify an instance by returning unique consecutive instance ids.
|
|
|
|
* It is reset when the first (primary) instance of your app starts and only incremented afterwards.
|
|
|
|
* @return Returns a unique instance id.
|
|
|
|
*/
|
2023-03-04 19:31:04 +01:00
|
|
|
quint32 SingleApplicationClass::instanceId() const {
|
|
|
|
|
|
|
|
#if defined(SINGLEAPPLICATION)
|
2022-02-06 00:07:15 +01:00
|
|
|
Q_D(const SingleApplication);
|
2023-03-04 19:31:04 +01:00
|
|
|
#elif defined(SINGLECOREAPPLICATION)
|
|
|
|
Q_D(const SingleCoreApplication);
|
|
|
|
#endif
|
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
return d->instanceNumber_;
|
2023-03-04 19:31:04 +01:00
|
|
|
|
2019-01-07 01:00:58 +01:00
|
|
|
}
|
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
/**
|
|
|
|
* Returns the OS PID (Process Identifier) of the process running the primary instance.
|
|
|
|
* Especially useful when SingleApplication is coupled with OS. specific APIs.
|
|
|
|
* @return Returns the primary instance PID.
|
|
|
|
*/
|
2023-03-04 19:31:04 +01:00
|
|
|
qint64 SingleApplicationClass::primaryPid() const {
|
|
|
|
|
|
|
|
#if defined(SINGLEAPPLICATION)
|
2022-02-06 00:07:15 +01:00
|
|
|
Q_D(const SingleApplication);
|
2023-03-04 19:31:04 +01:00
|
|
|
#elif defined(SINGLECOREAPPLICATION)
|
|
|
|
Q_D(const SingleCoreApplication);
|
|
|
|
#endif
|
|
|
|
|
2019-08-26 02:00:47 +02:00
|
|
|
return d->primaryPid();
|
2023-03-04 19:31:04 +01:00
|
|
|
|
2019-01-07 01:00:58 +01:00
|
|
|
}
|
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
/**
|
|
|
|
* Returns the username the primary instance is running as.
|
|
|
|
* @return Returns the username the primary instance is running as.
|
|
|
|
*/
|
2023-03-04 19:31:04 +01:00
|
|
|
QString SingleApplicationClass::primaryUser() const {
|
|
|
|
|
|
|
|
#if defined(SINGLEAPPLICATION)
|
2022-02-06 00:07:15 +01:00
|
|
|
Q_D(const SingleApplication);
|
2023-03-04 19:31:04 +01:00
|
|
|
#elif defined(SINGLECOREAPPLICATION)
|
|
|
|
Q_D(const SingleCoreApplication);
|
|
|
|
#endif
|
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
return d->primaryUser();
|
2023-03-04 19:31:04 +01:00
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the username the current instance is running as.
|
|
|
|
* @return Returns the username the current instance is running as.
|
|
|
|
*/
|
2023-03-04 19:31:04 +01:00
|
|
|
QString SingleApplicationClass::currentUser() const {
|
|
|
|
return SingleApplicationPrivateClass::getUsername();
|
2020-10-05 21:31:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sends message to the Primary Instance.
|
|
|
|
* @param message The message to send.
|
|
|
|
* @param timeout the maximum timeout in milliseconds for blocking functions.
|
2021-07-02 18:45:53 +02:00
|
|
|
* @return true if the message was sent successfully, false otherwise.
|
2020-10-05 21:31:03 +02:00
|
|
|
*/
|
2023-03-04 19:31:04 +01:00
|
|
|
bool SingleApplicationClass::sendMessage(const QByteArray &message, const int timeout) {
|
2019-08-26 02:00:47 +02:00
|
|
|
|
2023-03-04 19:31:04 +01:00
|
|
|
#if defined(SINGLEAPPLICATION)
|
2019-08-26 02:00:47 +02:00
|
|
|
Q_D(SingleApplication);
|
2023-03-04 19:31:04 +01:00
|
|
|
#elif defined(SINGLECOREAPPLICATION)
|
|
|
|
Q_D(SingleCoreApplication);
|
|
|
|
#endif
|
2019-08-26 02:00:47 +02:00
|
|
|
|
|
|
|
// Nobody to connect to
|
|
|
|
if (isPrimary()) return false;
|
2019-01-07 01:00:58 +01:00
|
|
|
|
2019-08-26 02:00:47 +02:00
|
|
|
// Make sure the socket is connected
|
2023-03-04 19:31:04 +01:00
|
|
|
if (!d->connectToPrimary(timeout, SingleApplicationPrivateClass::Reconnect)) {
|
2020-10-05 21:31:03 +02:00
|
|
|
return false;
|
2021-08-23 21:21:08 +02:00
|
|
|
}
|
2019-01-07 01:00:58 +01:00
|
|
|
|
2022-02-06 00:07:15 +01:00
|
|
|
return d->writeConfirmedMessage(timeout, message);
|
2023-03-04 19:31:04 +01:00
|
|
|
|
2020-10-05 21:31:03 +02:00
|
|
|
}
|