Add MDNS discovery
This commit is contained in:
parent
beb7e94cbb
commit
9e51b7d814
7 changed files with 202 additions and 9 deletions
|
@ -65,7 +65,7 @@ else()
|
|||
find_package(Qca-qt${QT_MAJOR_VERSION} ${QCA_MIN_VERSION} REQUIRED)
|
||||
set(Qca_LIBRARY qca-qt${QT_MAJOR_VERSION})
|
||||
|
||||
set(KF5_REQUIRED_COMPONENTS I18n ConfigWidgets DBusAddons IconThemes Notifications KIO KCMUtils Service Solid Kirigami2 People WindowSystem GuiAddons)
|
||||
set(KF5_REQUIRED_COMPONENTS I18n ConfigWidgets DBusAddons IconThemes Notifications KIO KCMUtils Service Solid Kirigami2 People WindowSystem GuiAddons DNSSD)
|
||||
set(KF5_OPTIONAL_COMPONENTS DocTools)
|
||||
|
||||
set_package_properties(KF5Kirigami2 PROPERTIES
|
||||
|
|
|
@ -64,6 +64,7 @@ PRIVATE
|
|||
Qt${QT_MAJOR_VERSION}::DBus
|
||||
KF5::I18n
|
||||
KF5::ConfigCore
|
||||
KF5::DNSSD
|
||||
)
|
||||
|
||||
if(${KF5KIO_FOUND})
|
||||
|
|
|
@ -8,6 +8,7 @@ set(backends_kdeconnect_SRCS
|
|||
backends/lan/compositeuploadjob.cpp
|
||||
backends/lan/uploadjob.cpp
|
||||
backends/lan/socketlinereader.cpp
|
||||
backends/lan/mdnsdiscovery.cpp
|
||||
|
||||
PARENT_SCOPE
|
||||
)
|
||||
|
|
|
@ -45,6 +45,7 @@ LanLinkProvider::LanLinkProvider(bool testMode, quint16 udpBroadcastPort, quint1
|
|||
, m_udpListenPort(udpListenPort)
|
||||
, m_testMode(testMode)
|
||||
, m_combineBroadcastsTimer(this)
|
||||
, m_mdnsDiscovery(this)
|
||||
{
|
||||
m_combineBroadcastsTimer.setInterval(0); // increase this if waiting a single event-loop iteration is not enough
|
||||
m_combineBroadcastsTimer.setSingleShot(true);
|
||||
|
@ -101,12 +102,17 @@ void LanLinkProvider::onStart()
|
|||
}
|
||||
}
|
||||
|
||||
onNetworkChange();
|
||||
broadcastUdpIdentityPacket();
|
||||
m_mdnsDiscovery.startAnnouncing();
|
||||
m_mdnsDiscovery.startDiscovering();
|
||||
|
||||
qCDebug(KDECONNECT_CORE) << "LanLinkProvider started";
|
||||
}
|
||||
|
||||
void LanLinkProvider::onStop()
|
||||
{
|
||||
m_mdnsDiscovery.stopAnnouncing();
|
||||
m_mdnsDiscovery.stopDiscovering();
|
||||
m_udpSocket.close();
|
||||
m_server->close();
|
||||
qCDebug(KDECONNECT_CORE) << "LanLinkProvider stopped";
|
||||
|
@ -125,15 +131,25 @@ void LanLinkProvider::onNetworkChange()
|
|||
void LanLinkProvider::broadcastToNetwork()
|
||||
{
|
||||
if (!m_server->isListening()) {
|
||||
// Not started
|
||||
qWarning() << "TCP server not listening, not broadcasting";
|
||||
return;
|
||||
}
|
||||
|
||||
Q_ASSERT(m_tcpPort != 0);
|
||||
|
||||
qCDebug(KDECONNECT_CORE()) << "Broadcasting identity packet";
|
||||
broadcastUdpIdentityPacket();
|
||||
m_mdnsDiscovery.stopDiscovering();
|
||||
m_mdnsDiscovery.startDiscovering();
|
||||
}
|
||||
|
||||
QList<QHostAddress> destinations = getBroadcastAddresses();
|
||||
void LanLinkProvider::broadcastUdpIdentityPacket()
|
||||
{
|
||||
sendUdpIdentityPacket(getBroadcastAddresses());
|
||||
}
|
||||
|
||||
void LanLinkProvider::sendUdpIdentityPacket(const QList<QHostAddress> &destinations)
|
||||
{
|
||||
qCDebug(KDECONNECT_CORE()) << "Broadcasting identity packet";
|
||||
|
||||
NetworkPacket np = KdeConnectConfig::instance().deviceInfo().toIdentityPacket();
|
||||
np.set(QStringLiteral("tcpPort"), m_tcpPort);
|
||||
|
@ -175,14 +191,14 @@ void LanLinkProvider::broadcastToNetwork()
|
|||
if (sourceAddress.protocol() == QAbstractSocket::IPv4Protocol && sourceAddress != QHostAddress::LocalHost) {
|
||||
qCDebug(KDECONNECT_CORE()) << "Broadcasting as" << sourceAddress;
|
||||
sendSocket.bind(sourceAddress);
|
||||
sendBroadcasts(sendSocket, np, destinations);
|
||||
sendUdpPacket(sendSocket, np, destinations);
|
||||
sendSocket.close();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
#else
|
||||
sendBroadcasts(m_udpSocket, np, destinations);
|
||||
sendUdpPacket(m_udpSocket, np, destinations);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
@ -209,7 +225,7 @@ QList<QHostAddress> LanLinkProvider::getBroadcastAddresses()
|
|||
return destinations;
|
||||
}
|
||||
|
||||
void LanLinkProvider::sendBroadcasts(QUdpSocket &socket, const NetworkPacket &np, const QList<QHostAddress> &addresses)
|
||||
void LanLinkProvider::sendUdpPacket(QUdpSocket &socket, const NetworkPacket &np, const QList<QHostAddress> &addresses)
|
||||
{
|
||||
const QByteArray payload = np.serialize();
|
||||
|
||||
|
|
|
@ -17,6 +17,7 @@
|
|||
#include "backends/linkprovider.h"
|
||||
#include "kdeconnectcore_export.h"
|
||||
#include "landevicelink.h"
|
||||
#include "mdnsdiscovery.h"
|
||||
#include "server.h"
|
||||
|
||||
class KDECONNECTCORE_EXPORT LanLinkProvider : public LinkProvider
|
||||
|
@ -37,6 +38,13 @@ public:
|
|||
return QStringLiteral("LanLinkProvider");
|
||||
}
|
||||
|
||||
void sendUdpIdentityPacket(const QList<QHostAddress> &addresses);
|
||||
|
||||
QHostAddress localAddress() const
|
||||
{
|
||||
return m_udpSocket.localAddress();
|
||||
};
|
||||
|
||||
static void configureSslSocket(QSslSocket *socket, const QString &deviceId, bool isDeviceTrusted);
|
||||
static void configureSocket(QSslSocket *socket);
|
||||
|
||||
|
@ -67,7 +75,8 @@ private:
|
|||
void onNetworkConfigurationChanged(const QNetworkConfiguration &config);
|
||||
void addLink(QSslSocket *socket, const DeviceInfo &deviceInfo);
|
||||
QList<QHostAddress> getBroadcastAddresses();
|
||||
void sendBroadcasts(QUdpSocket &socket, const NetworkPacket &np, const QList<QHostAddress> &addresses);
|
||||
void sendUdpPacket(QUdpSocket &socket, const NetworkPacket &np, const QList<QHostAddress> &addresses);
|
||||
void broadcastUdpIdentityPacket();
|
||||
|
||||
Server *m_server;
|
||||
QUdpSocket m_udpSocket;
|
||||
|
@ -86,6 +95,8 @@ private:
|
|||
QNetworkConfiguration m_lastConfig;
|
||||
const bool m_testMode;
|
||||
QTimer m_combineBroadcastsTimer;
|
||||
|
||||
MdnsDiscovery m_mdnsDiscovery;
|
||||
};
|
||||
|
||||
#endif
|
||||
|
|
122
core/backends/lan/mdnsdiscovery.cpp
Normal file
122
core/backends/lan/mdnsdiscovery.cpp
Normal file
|
@ -0,0 +1,122 @@
|
|||
/**
|
||||
* SPDX-FileCopyrightText: 2023 Albert Vaca <albertvaka@gmail.com>
|
||||
*
|
||||
* SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
|
||||
*/
|
||||
|
||||
#include "mdnsdiscovery.h"
|
||||
|
||||
#include "core_debug.h"
|
||||
#include "kdeconnectconfig.h"
|
||||
#include "lanlinkprovider.h"
|
||||
|
||||
#include <KDNSSD/DNSSD/PublicService>
|
||||
#include <KDNSSD/DNSSD/RemoteService>
|
||||
#include <KDNSSD/DNSSD/ServiceBrowser>
|
||||
|
||||
const QString kServiceName = QStringLiteral("_kdeconnect._udp");
|
||||
|
||||
MdnsDiscovery::MdnsDiscovery(LanLinkProvider *lanLinkProvider)
|
||||
: lanLinkProvider(lanLinkProvider)
|
||||
{
|
||||
switch (KDNSSD::ServiceBrowser::isAvailable()) {
|
||||
case KDNSSD::ServiceBrowser::Stopped:
|
||||
qWarning() << "mDNS or Avahi daemons are not running, mDNS discovery not available";
|
||||
break;
|
||||
case KDNSSD::ServiceBrowser::Working:
|
||||
qCDebug(KDECONNECT_CORE) << "mDNS discovery is available";
|
||||
break;
|
||||
case KDNSSD::ServiceBrowser::Unsupported:
|
||||
qWarning() << "mDNS discovery not available (library built without DNS-SD support)";
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
MdnsDiscovery::~MdnsDiscovery()
|
||||
{
|
||||
stopAnnouncing();
|
||||
stopDiscovering();
|
||||
}
|
||||
|
||||
void MdnsDiscovery::startAnnouncing()
|
||||
{
|
||||
if (m_publisher != nullptr) {
|
||||
qCDebug(KDECONNECT_CORE) << "MDNS already announcing";
|
||||
return;
|
||||
}
|
||||
qCDebug(KDECONNECT_CORE) << "MDNS start announcing";
|
||||
|
||||
KdeConnectConfig &config = KdeConnectConfig::instance();
|
||||
|
||||
m_publisher = new KDNSSD::PublicService(config.deviceId(), kServiceName, LanLinkProvider::UDP_PORT, QStringLiteral("local"));
|
||||
m_publisher->setParent(this);
|
||||
|
||||
// We can't fit the device certificate in this field, so this is not enough info to create a Device and won't be used.
|
||||
QMap<QString, QByteArray> data;
|
||||
data[QStringLiteral("id")] = config.deviceId().toUtf8();
|
||||
data[QStringLiteral("name")] = config.name().toUtf8();
|
||||
data[QStringLiteral("type")] = config.deviceType().toString().toUtf8();
|
||||
data[QStringLiteral("protocol")] = QString::number(NetworkPacket::s_protocolVersion).toUtf8();
|
||||
data[QStringLiteral("port")] = QString::number(LanLinkProvider::UDP_PORT).toUtf8(); // iOS needs ip and port here
|
||||
data[QStringLiteral("ip")] = lanLinkProvider->localAddress().toString().toUtf8();
|
||||
m_publisher->setTextData(data);
|
||||
|
||||
connect(m_publisher, &KDNSSD::PublicService::published, [](bool successful) {
|
||||
if (successful) {
|
||||
qCDebug(KDECONNECT_CORE) << "MDNS published successfully";
|
||||
} else {
|
||||
qWarning() << "MDNS failed to publish";
|
||||
}
|
||||
});
|
||||
|
||||
m_publisher->publishAsync();
|
||||
}
|
||||
|
||||
void MdnsDiscovery::stopAnnouncing()
|
||||
{
|
||||
if (m_publisher != nullptr) {
|
||||
qCDebug(KDECONNECT_CORE) << "MDNS stop announcing";
|
||||
delete m_publisher;
|
||||
m_publisher = nullptr;
|
||||
}
|
||||
}
|
||||
|
||||
void MdnsDiscovery::startDiscovering()
|
||||
{
|
||||
if (m_serviceBrowser != nullptr) {
|
||||
qCDebug(KDECONNECT_CORE) << "MDNS already discovering";
|
||||
return;
|
||||
}
|
||||
qCDebug(KDECONNECT_CORE) << "MDNS start discovering";
|
||||
|
||||
m_serviceBrowser = new KDNSSD::ServiceBrowser(kServiceName, true);
|
||||
|
||||
connect(m_serviceBrowser, &KDNSSD::ServiceBrowser::serviceAdded, [this](KDNSSD::RemoteService::Ptr service) {
|
||||
if (KdeConnectConfig::instance().deviceId() == service->serviceName()) {
|
||||
qCDebug(KDECONNECT_CORE) << "Discovered myself, ignoring";
|
||||
return;
|
||||
}
|
||||
qCDebug(KDECONNECT_CORE) << "Discovered " << service->serviceName() << " at " << service->hostName();
|
||||
QHostAddress address(service->hostName());
|
||||
lanLinkProvider->sendUdpIdentityPacket(QList<QHostAddress>{address});
|
||||
});
|
||||
|
||||
connect(m_serviceBrowser, &KDNSSD::ServiceBrowser::serviceRemoved, [](KDNSSD::RemoteService::Ptr service) {
|
||||
qCDebug(KDECONNECT_CORE) << "Lost " << service->serviceName();
|
||||
});
|
||||
|
||||
connect(m_serviceBrowser, &KDNSSD::ServiceBrowser::finished, []() {
|
||||
qCDebug(KDECONNECT_CORE) << "Finished discovery";
|
||||
});
|
||||
|
||||
m_serviceBrowser->startBrowse();
|
||||
}
|
||||
|
||||
void MdnsDiscovery::stopDiscovering()
|
||||
{
|
||||
if (m_serviceBrowser != nullptr) {
|
||||
qCDebug(KDECONNECT_CORE) << "MDNS stop discovering";
|
||||
delete m_serviceBrowser;
|
||||
m_serviceBrowser = nullptr;
|
||||
}
|
||||
}
|
42
core/backends/lan/mdnsdiscovery.h
Normal file
42
core/backends/lan/mdnsdiscovery.h
Normal file
|
@ -0,0 +1,42 @@
|
|||
/**
|
||||
* SPDX-FileCopyrightText: 2023 Albert Vaca Cintora <albertvaka@gmail.com>
|
||||
*
|
||||
* SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
|
||||
*/
|
||||
|
||||
#ifndef KDECONNECT_MDNS_DISCOVERY_H
|
||||
#define KDECONNECT_MDNS_DISCOVERY_H
|
||||
|
||||
#include <QObject>
|
||||
|
||||
#include "kdeconnectcore_export.h"
|
||||
|
||||
class LanLinkProvider;
|
||||
namespace KDNSSD
|
||||
{
|
||||
class PublicService;
|
||||
class ServiceBrowser;
|
||||
};
|
||||
|
||||
// This class overrides QTcpServer to bind QSslSocket to native socket descriptor instead of QTcpSocket
|
||||
class KDECONNECTCORE_EXPORT MdnsDiscovery : public QObject
|
||||
{
|
||||
Q_OBJECT
|
||||
|
||||
public:
|
||||
MdnsDiscovery(LanLinkProvider *parent);
|
||||
~MdnsDiscovery();
|
||||
|
||||
void startDiscovering();
|
||||
void stopDiscovering();
|
||||
|
||||
void stopAnnouncing();
|
||||
void startAnnouncing();
|
||||
|
||||
private:
|
||||
LanLinkProvider *lanLinkProvider = nullptr;
|
||||
KDNSSD::PublicService *m_publisher = nullptr;
|
||||
KDNSSD::ServiceBrowser *m_serviceBrowser = nullptr;
|
||||
};
|
||||
|
||||
#endif // KDECONNECT_SERVER_H
|
Loading…
Reference in a new issue