/* * Deskflow -- mouse and keyboard sharing utility * SPDX-FileCopyrightText: (C) 2025 Symless Ltd. * SPDX-License-Identifier: GPL-2.0-only WITH LicenseRef-OpenSSL-Exception */ #include "DaemonIpcClient.h" #include "common/Constants.h" #include #include #include #include namespace deskflow::gui::ipc { const auto kTimeout = 1000; const auto kRetryLimit = 3; DaemonIpcClient::DaemonIpcClient(QObject *parent) : QObject(parent), m_socket{new QLocalSocket(this)} // NOSONAR - Qt memory { connect(m_socket, &QLocalSocket::disconnected, this, &DaemonIpcClient::handleDisconnected); connect(m_socket, &QLocalSocket::errorOccurred, this, &DaemonIpcClient::handleErrorOccurred); } bool DaemonIpcClient::connectToServer() { if (m_state == State::Connecting) { qWarning() << "daemon ipc client already connecting to server"; return false; } if (m_state != State::Unconnected) { qDebug() << "daemon ipc client not in unconnected state, disconnecting"; disconnectFromServer(); } if (m_socket->state() != QLocalSocket::UnconnectedState) { qWarning() << "daemon ipc client socket not in unconnected state, disconnecting"; disconnectFromServer(); } for (int i = 0; i < kRetryLimit; ++i) { if (i == 0) { qDebug() << "daemon ipc client connecting to server:" << kDaemonIpcName; } else { qDebug() << "daemon ipc client retrying connection, attempt:" << i + 1; } m_state = State::Connecting; m_socket->connectToServer(kDaemonIpcName); if (!m_socket->waitForConnected(kTimeout)) { qWarning() << "daemon ipc client failed to connect"; disconnectFromServer(); continue; } if (!sendMessage("hello", "hello", false)) { qWarning() << "daemon ipc client failed to send hello"; disconnectFromServer(); continue; } m_state = State::Connected; qDebug() << "daemon ipc client connected"; Q_EMIT connected(); return true; } qWarning() << "daemon ipc client failed to connect after" << kRetryLimit << "attempts"; disconnectFromServer(); Q_EMIT connectionFailed(); return false; } void DaemonIpcClient::disconnectFromServer() { m_state = State::Disconnecting; qDebug() << "daemon ipc client disconnecting from server"; m_socket->disconnectFromServer(); if (m_socket->state() != QLocalSocket::UnconnectedState) { qDebug() << "daemon ipc client waiting for socket to disconnect"; m_socket->waitForDisconnected(kTimeout); qDebug() << "daemon ipc client disconnected from server"; } else { qDebug() << "daemon ipc client socket already disconnected"; } m_state = State::Unconnected; } void DaemonIpcClient::handleDisconnected() { qDebug() << "daemon ipc client disconnected from server"; if (m_state == State::Connected) { Q_EMIT connectionFailed(); } m_state = State::Unconnected; } void DaemonIpcClient::handleErrorOccurred() { qWarning() << "daemon ipc client error:" << m_socket->errorString(); disconnectFromServer(); if (m_state == State::Connected) { Q_EMIT connectionFailed(); } } bool DaemonIpcClient::sendMessage(const QString &message, const QString &expectAck, const bool expectConnected) { if (expectConnected && !isConnected()) { qWarning() << "cannot send command, ipc client not connected"; return false; } QByteArray messageData = message.toUtf8() + "\n"; m_socket->write(messageData); if (!m_socket->waitForBytesWritten(kTimeout)) { qWarning() << "daemon ipc client failed to write command"; return false; } if (!expectAck.isEmpty()) { qDebug() << "daemon ipc client waiting for ack: " << expectAck; if (!m_socket->waitForReadyRead(kTimeout)) { qWarning() << "daemon ipc client socket ready read timed out"; return false; } QByteArray response = m_socket->readAll(); if (response.isEmpty()) { qWarning() << "daemon ipc client got empty response"; return false; } QString responseData = QString::fromUtf8(response); if (responseData.isEmpty()) { qWarning() << "daemon ipc client failed to convert response to string"; return false; } if (responseData != expectAck + "\n") { qWarning() << "daemon ipc client got unexpected response: " << responseData; return false; } } qDebug() << "daemon ipc client sent message: " << messageData; return true; } bool DaemonIpcClient::keepAlive() { if (!isConnected() && !connectToServer()) { qWarning() << "daemon ipc client keep alive failed to connect"; return false; } if (!sendMessage("noop")) { qWarning() << "daemon ipc client keep alive ping failed, reconnecting"; connectToServer(); return false; } return true; } bool DaemonIpcClient::sendLogLevel(const QString &logLevel) { if (!keepAlive()) return false; sendMessage("logLevel=" + logLevel); return true; } bool DaemonIpcClient::sendStartProcess(const QString &command, bool elevate) { if (!keepAlive()) return false; if (!sendMessage("elevate=" + (elevate ? QStringLiteral("yes") : QStringLiteral("no")))) { return false; } if (!sendMessage("command=" + command)) { return false; } return sendMessage("start"); } bool DaemonIpcClient::sendStopProcess() { return sendMessage("stop"); } QString DaemonIpcClient::requestLogPath() { if (!keepAlive()) return QString(); if (!sendMessage("logPath", QString())) { return QString(); } if (!m_socket->waitForReadyRead(kTimeout)) { qWarning() << "daemon ipc client failed to read log path response"; return QString(); } QByteArray response = m_socket->readAll(); if (response.isEmpty()) { qWarning() << "daemon ipc client got empty log path response"; return QString(); } QString responseData = QString::fromUtf8(response); if (responseData.isEmpty()) { qWarning() << "daemon ipc client failed to convert log path response to string"; return QString(); } // Trimming removes newline from end of message. QStringList parts = responseData.trimmed().split("="); if (parts.size() != 2) { qWarning() << "daemon ipc client got invalid log path response: " << responseData; return QString(); } if (parts[0] != "logPath") { qWarning() << "daemon ipc client got unexpected log path response: " << responseData; return QString(); } return parts[1]; } bool DaemonIpcClient::sendClearSettings() { return sendMessage("clearSettings"); } } // namespace deskflow::gui::ipc