/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Copyright 2013 - 2020, nymea GmbH * Contact: contact@nymea.io * * This file is part of nymea. * This project including source code and documentation is protected by * copyright law, and remains the property of nymea GmbH. All rights, including * reproduction, publication, editing and translation, are reserved. The use of * this project is subject to the terms of a license agreement to be concluded * with nymea GmbH in accordance with the terms of use of nymea GmbH, available * under https://nymea.io/license * * GNU General Public License Usage * Alternatively, this project may be redistributed and/or modified under the * terms of the GNU General Public License as published by the Free Software * Foundation, GNU version 3. This project 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 project. If not, see . * * For any further details and any questions please contact us under * contact@nymea.io or see our FAQ/Licensing Information on * https://nymea.io/license/faq * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ #include "zeroconfdiscovery.h" #include #include "../nymeahost.h" #include "logging.h" NYMEA_LOGGING_CATEGORY(dcZeroConf, "ZeroConf") ZeroconfDiscovery::ZeroconfDiscovery(NymeaHosts *nymeaHosts, QObject *parent) : QObject(parent), m_nymeaHosts(nymeaHosts) { #ifdef WITH_ZEROCONF // NOTE: There seem to be too many issues in QtZeroConf and IPv6. // See https://github.com/jbagg/QtZeroConf/issues/22 // Limiting this to IPv4 for now... m_zeroconfJsonRPC = new QZeroConf(this); connect(m_zeroconfJsonRPC, &QZeroConf::serviceAdded, this, &ZeroconfDiscovery::serviceEntryAdded); connect(m_zeroconfJsonRPC, &QZeroConf::serviceUpdated, this, &ZeroconfDiscovery::serviceEntryAdded); connect(m_zeroconfJsonRPC, &QZeroConf::serviceRemoved, this, &ZeroconfDiscovery::serviceEntryRemoved); if (m_zeroconfJsonRPC->isValid()) { m_zeroconfJsonRPC->startBrowser("_jsonrpc._tcp", QAbstractSocket::IPv4Protocol); qCInfo(dcZeroConf()) << "Created service browser for _jsonrpc._tcp:" << m_zeroconfJsonRPC->browserExists(); } else { qCWarning(dcZeroConf()) << "Failed to initialize service broeser for _jsonprc._tcp"; } m_zeroconfWebSocket = new QZeroConf(this); connect(m_zeroconfWebSocket, &QZeroConf::serviceAdded, this, &ZeroconfDiscovery::serviceEntryAdded); connect(m_zeroconfWebSocket, &QZeroConf::serviceUpdated, this, &ZeroconfDiscovery::serviceEntryAdded); connect(m_zeroconfWebSocket, &QZeroConf::serviceRemoved, this, &ZeroconfDiscovery::serviceEntryRemoved); if (m_zeroconfWebSocket->isValid()) { m_zeroconfWebSocket->startBrowser("_ws._tcp", QAbstractSocket::IPv4Protocol); qCInfo(dcZeroConf()) << "Created service browser for _ws._tcp:" << m_zeroconfWebSocket->browserExists(); } else { qCWarning(dcZeroConf()) << "Failed to initialize service browserr for _ws._tcp"; } #else qCInfo(dcZeroConf()) << "Zeroconf support not compiled in. Zeroconf will not be available."; #endif } ZeroconfDiscovery::~ZeroconfDiscovery() { qCInfo(dcZeroConf()) << "Shutting down service browsers"; } bool ZeroconfDiscovery::available() const { #ifdef WITH_ZEROCONF return m_zeroconfJsonRPC->browserExists() || m_zeroconfWebSocket->browserExists(); #else return false; #endif } bool ZeroconfDiscovery::discovering() const { return available(); } #ifdef WITH_ZEROCONF void ZeroconfDiscovery::serviceEntryAdded(const QZeroConfService &entry) { if (!entry->name().startsWith("nymea")) { // Skip non-nymea services altogether qCDebug(dcZeroConf()) << "Skipping service entry:" << entry << entry->ip() << entry->txt() << entry->type(); return; } if (entry->ip().isNull()) { // Skip entries that don't have an ip address at all for some reason qCDebug(dcZeroConf()) << "Skipping service entry:" << entry << entry->ip() << entry->txt() << entry->type(); return; } if (entry->ip().toString().startsWith("fe80")) { // Skip link-local-IPv6 results qCDebug(dcZeroConf()) << "Skipping service entry:" << entry << entry->ip() << entry->txt() << entry->type(); return; } qCDebug(dcZeroConf()) << "Service discovered" << entry->type() << entry->name() << " IP:" << entry->ip().toString() << entry->txt(); QString uuid; bool sslEnabled = false; QString serverName; QString version; foreach (const QByteArray &key, entry->txt().keys()) { QPair txtRecord = qMakePair(key, entry->txt().value(key)); if (!sslEnabled && txtRecord.first == "sslEnabled") { sslEnabled = (txtRecord.second == "true"); } if (txtRecord.first == "uuid") { uuid = txtRecord.second; } if (txtRecord.first == "name") { serverName = txtRecord.second; } if (txtRecord.first == "serverVersion") { version = txtRecord.second; } } qCDebug(dcZeroConf()) << "Service entry added" << serverName << uuid << sslEnabled; NymeaHost* host = m_nymeaHosts->find(uuid); if (!host) { host = new NymeaHost(m_nymeaHosts); host->setUuid(uuid); qCInfo(dcZeroConf()) << "Adding new host:" << serverName << uuid; m_nymeaHosts->addHost(host); } host->setName(serverName); host->setVersion(version); QUrl url; // NOTE: On linux this is "_jsonrpc._tcp" while on apple systems this is "_jsonrpc._tcp." if (entry->type().startsWith("_jsonrpc._tcp")) { url.setScheme(sslEnabled ? "nymeas" : "nymea"); } else if (entry->type().startsWith("_ws._tcp")) { url.setScheme(sslEnabled ? "wss" : "ws"); } url.setHost(entry->ip().toString()); url.setPort(entry->port()); Connection *connection = host->connections()->find(url); if (!connection) { qCInfo(dcZeroConf()) << "Adding new connection to host:" << host->name() << url.toString(); QString displayName = QString("%1:%2").arg(url.host()).arg(url.port()); Connection::BearerType bearerType = QHostAddress(url.host()).isLoopback() ? Connection::BearerTypeLoopback : Connection::BearerTypeLan; connection = new Connection(url, bearerType, sslEnabled, displayName); connection->setOnline(true); host->connections()->addConnection(connection); } else { qCInfo(dcZeroConf()) << "Setting connection online:" << host->name() << url.toString(); connection->setOnline(true); } } void ZeroconfDiscovery::serviceEntryRemoved(const QZeroConfService &entry) { if (!entry->name().startsWith("nymea") || entry->ip().isNull()) { return; } QString uuid; bool sslEnabled = false; QString serverName; QString version; foreach (const QByteArray &key, entry->txt().keys()) { QPair txtRecord = qMakePair(key, entry->txt().value(key)); if (!sslEnabled && txtRecord.first == "sslEnabled") { sslEnabled = (txtRecord.second == "true"); } if (txtRecord.first == "uuid") { uuid = txtRecord.second; } if (txtRecord.first == "name") { serverName = txtRecord.second; } if (txtRecord.first == "serverVersion") { version = txtRecord.second; } } qCDebug(dcZeroConf()) << "Service entry removed" << entry->name(); NymeaHost* host = m_nymeaHosts->find(uuid); if (!host) { // Nothing to do... return; } QUrl url; if (entry->type() == "_jsonrpc._tcp") { url.setScheme(sslEnabled ? "nymeas" : "nymea"); } else { url.setScheme(sslEnabled ? "wss" : "ws"); } url.setHost(entry->ip().toString()); url.setPort(entry->port()); Connection *connection = host->connections()->find(url); if (!connection){ // Connection url not found... return; } qCInfo(dcZeroConf()) << "Setting connection offline:" << host->name() << url.toString(); connection->setOnline(false); } #endif