KDBusService Class

KDBusService takes care of registering the current process with D-Bus. More...

Header: #include <KDBusService>
CMake: find_package(KF6 REQUIRED COMPONENTS DBusAddons)
target_link_libraries(mytarget PRIVATE KF6::DBusAddons)
Since: 5.0
Inherits: QObject

Public Types

enum StartupOption { Unique, Multiple, NoExitOnFailure, Replace }
flags StartupOptions

Public Functions

KDBusService(KDBusService::StartupOptions options = Multiple, QObject *parent = nullptr)
virtual ~KDBusService() override
QString errorMessage() const
bool isRegistered() const
(since 5.33) QString serviceName() const
void setExitValue(int value)

Public Slots

void unregister()

Signals

void activateActionRequested(const QString &actionName, const QVariant &parameter)
void activateRequested(const QStringList &arguments, const QString &workingDirectory)
void openRequested(const QList<QUrl> &uris)

Detailed Description

This registers the application at a predictable location on D-Bus, registers the QCoreApplication (or subclass) object at /MainApplication, and assists in implementing the application side of D-Bus activation from the Desktop Entry Specification.

An application can either work in Multiple mode or Unique mode.

In Multiple mode, the application can be launched many times. The service name in the D-Bus registration will contain the PID to distinguish the various instances; for example: org.kde.konqueror-12345.

In Unique mode, only one instance of this application can ever run. The first instance of the application registers with D-Bus without the PID, and any attempt to run the application again will cause the activateRequested() signal to be emitted in the already-running instance; the duplicate instance will then quit. The exit value can be set by the already running instance with setExitValue(), the default value is 0.

Unique-mode applications should usually delay parsing command-line arguments until after creating a KDBusService object; that way they know they are the original instance of the application.

Applications that set the D-Bus activation entry (DBusActivatable=true) in their desktop files will use Unique mode and connect to the signals emitted by this class. Note that the D-Bus interface is exported for Multiple-mode applications as well, so it also makes sense for such applications to connect to the signals emitted by this class.

Note: In order to avoid a race, the application should export its objects to D-Bus before allowing the event loop to run (for example, by calling QCoreApplication::exec()). Otherwise, the application will appear on the bus before its objects are accessible via D-Bus, which could be a problem for other applications or scripts which start the application in order to talk D-Bus to it immediately.

Example usage:

QApplication app(argc, argv);
app.setApplicationName("kuiserver");
app.setOrganizationDomain("kde.org");
// Create your D-Bus objects here
// ...
KDBusService service(KDBusService::Unique);
// If this point is reached, this is the only running instance
// i.e. org.kde.kuiserver has been registered
return app.exec();

Member Type Documentation

enum KDBusService::StartupOption
flags KDBusService::StartupOptions

Options to control the behaviour of KDBusService

ConstantValueDescription
KDBusService::Unique1Indicates that only one instance of this application should ever exist. Cannot be combined with Multiple.
KDBusService::Multiple2Indicates that multiple instances of the application may exist. Cannot be combined with Unique. This is the default.
KDBusService::NoExitOnFailure4Indicates that the application should not exit if it failed to register with D-Bus. If not set, KDBusService will quit the application if it failed to register the service with D-Bus or a Unique instance can not be activated. A Multiple instance will exit with error code 1. The exit value of a Unique instance can be set from the running instance with setExitValue(), the default value is 0.
KDBusService::Replace (since 5.65)8Indicates that if there's already a unique service running, to be quit and replaced with our own. If exported, it will try first quitting the service calling org.qtproject.Qt.QCoreApplication.quit, which is exported by KDBusService by default.

The StartupOptions type is a typedef for QFlags<StartupOption>. It stores an OR combination of StartupOption values.

Member Function Documentation

[explicit] KDBusService::KDBusService(KDBusService::StartupOptions options = Multiple, QObject *parent = nullptr)

Tries to register the current process to D-Bus at an address based on the application name and organization domain under the given parent.

The D-Bus service name is the reversed organization domain, followed by the application name. If options includes the Multiple flag, the application PID will be appended. For example:

app.setApplicationName("kuiserver");
app.setOrganizationDomain("kde.org");

The above will make KDBusService register as org.kde.kuiserver in Unique mode, and org.kde.kuiserver-1234 (if the process has PID 1234) in Multiple mode.

[override virtual noexcept] KDBusService::~KDBusService()

Destroys this object (but does not unregister the application).

Deleting this object before unregister() could confuse clients, who will see the service on the bus but will be unable to use the activation methods.

[signal] void KDBusService::activateActionRequested(const QString &actionName, const QVariant &parameter)

Signals that an application action actionName should be triggered with the given parameter.

This signal is emitted to request handling of the respective method of the D-Bus activation. For GUI applications, the signal handler also needs to deal with any platform-specific startup ids and make sure the mainwindow is shown as well as request its activation from the window manager. See documentation of activateRequested(const QStringList &arguments, const QString &) for details.

See the desktop entry specification for more information about action activation.

[signal] void KDBusService::activateRequested(const QStringList &arguments, const QString &workingDirectory)

Signals that the application is to be activated.

If this is a Unique application, when KDBusService is constructed in subsequent instances of the application (ie: when the executable is run when an instance is already running), it will cause this signal to be emitted in the already-running instance (with the arguments passed to the duplicate instance), and the duplicate instance will then exit.

If this application's desktop file indicates that it supports D-Bus activation (DBusActivatable=true), a command launcher may also call the Activate() D-Bus method to trigger this signal. In this case, arguments will be empty.

In single-window applications, the connected signal should typically raise the window.

arguments The arguments the executable was called with, starting with the executable file name. See QCoreApplication::arguments(). This can be empty.

workingDirectory The directory from which the executable is called. This can be empty.

A typical implementation of the signal handler would be:

if (!arguments.isEmpty()) {
    commandLineParser->parse(arguments); // same QCommandLineParser instance as the one used in main()
    handleCmdLine(workingDirectory); // shared method with main(), which uses commandLineParser to handle options and positional arguments
}

For GUI applications, the handler also needs to deal with any platform-specific startup ids and make sure the mainwindow is shown as well as request its activation from the window manager. For X11, KDBusService makes the id for the Startup Notification protocol available from QX11Info::nextStartupId(), if there is one. For Wayland, KDBusService provides the token for the XDG Activation protocol in the "XDG_ACTIVATION_TOKEN" environment variable and unsets it again after the signal, if there is one. The util method KWindowSystem::updateStartupId(QWindow *window) (since KF 5.91) takes care of that. A typical implementation in the signal handler would be:

mainWindow->show();
KWindowSystem::updateStartupId(mainWindow->windowHandle());
mainWindow->raise();
KWindowSystem::activateWindow(mainWindow->windowHandle());

If you're using the builtin handling of --help and --version in QCommandLineParser, you should call parser.process(arguments) before creating the KDBusService instance, since parse() doesn't handle those (and exiting the already-running instance would be wrong anyway).

See also setExitValue().

QString KDBusService::errorMessage() const

Returns the error message from the D-Bus registration if it failed.

Note that this is only useful when specifying the option NoExitOnFailure. Otherwise the process has quit by the time you can get a chance to call this.

bool KDBusService::isRegistered() const

Returns true if the D-Bus registration succeeded.

Note that this is only useful when specifying the option NoExitOnFailure. Otherwise, the simple fact that this process is still running indicates that the registration succeeded.

[signal] void KDBusService::openRequested(const QList<QUrl> &uris)

Signals that one or more files should be opened in the application.

This signal is emitted to request handling of the respective method of the D-Bus activation. For GUI applications, the signal handler also needs to deal with any platform-specific startup ids and make sure the mainwindow is shown as well as request its activation from the window manager. See documentation of activateRequested(const QStringList &arguments, const QString &) for details.

uris The URLs of the files to open.

[since 5.33] QString KDBusService::serviceName() const

Returns the name of the D-Bus service registered by this class. Mostly useful when using the option Multiple.

This function was introduced in 5.33.

void KDBusService::setExitValue(int value)

Sets the exit value to be used for a duplicate instance.

If this is a Unique application, a slot connected to activateRequested() can use this to specify a non-zero exit value for the duplicate instance. This would typically be done if invalid command-line arguments are passed.

Note that this will only work if the signal-slot connection type is Qt::DirectConnection.

value The exit value for the duplicate instance.

[slot] void KDBusService::unregister()

Manually unregister the given serviceName from D-Bus.