/****************************************************************************** ** ** Copyright (C) 2018 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtKnx module. ** ** $QT_BEGIN_LICENSE:GPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 or (at your option) any later version ** approved by the KDE Free Qt Foundation. The licenses are as published by ** the Free Software Foundation and appearing in the file LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ******************************************************************************/ #include "qknxnetipserverdescriptionagent.h" #include "qknxnetipserverdescriptionagent_p.h" QT_BEGIN_NAMESPACE /*! \class QKnxNetIpServerDescriptionAgent \inmodule QtKnx \ingroup qtknx-netip \brief The QKnxNetIpServerDescriptionAgent class establishes a point-to-point connection with a KNXnet/IP server and requests its description. First, the \l QKnxNetIpServerDiscoveryAgent class is used to search the network for available KNXnet/IP devices. After successful discovery of such a device, the server needs to be chosen to fetch the description. This can be been done as illustrated by the following code snippet: \code QKnxNetIpServerDescriptionAgent descriptionAgent; QHostAddress clientLocalAddress = ... descriptionAgent.setLocalAddress(clientLocalAddress); auto server = discoveryAgent.discoveredServers()[0]; // for example descriptionAgent.start(server); \endcode When the description is received from the server, the \l descriptionReceived() signal is emitted. The description received through this point-to-point connection may be more complete than the one sent during discovery. \sa {Qt KNXnet/IP Connection Classes} */ /*! \enum QKnxNetIpServerDescriptionAgent::State This enum value holds the state of the description agent. \value NotRunning The description agent is not running. \value Starting The description agent is starting up. \value Running The description agent is running. \value Stopping The description agent is stopping. */ /*! \enum QKnxNetIpServerDescriptionAgent::Error This enum value holds the type of an error that occurred. \value None No errors occurred. \value Network A network error occurred. \value NotIPv4 The network protocol used is not IPv4. \value Timeout A timeout occurred while waiting for the description response. \value Unknown An unknown error occurred. */ /*! \fn QKnxNetIpServerDescriptionAgent::descriptionReceived(QKnxNetIpServerInfo server) This signal is emitted when the description of \a server is received. */ /*! \fn QKnxNetIpServerDescriptionAgent::errorOccurred(QKnxNetIpServerDescriptionAgent::Error error, QString errorString) This signal is emitted when the error \a error with the message \a errorString occurs. */ /*! \fn QKnxNetIpServerDescriptionAgent::finished() This signal is emitted when the description agent has finished. */ /*! \fn QKnxNetIpServerDescriptionAgent::started() This signal is emitted when the description agent starts. */ /*! \fn QKnxNetIpServerDescriptionAgent::stateChanged(QKnxNetIpServerDescriptionAgent::State state) This signal is emitted when the state of the description agent changes to \a state. */ // -- QKnxNetIpServerDescriptionAgentPrivate QKnxNetIpServerDescriptionAgentPrivate::QKnxNetIpServerDescriptionAgentPrivate( const QHostAddress &addr, quint16 prt) : port(prt) , address(addr) {} namespace QKnxPrivate { static void clearSocket(QUdpSocket **socket) { if (*socket) { (*socket)->disconnect(); (*socket)->deleteLater(); (*socket) = nullptr; } } } void QKnxNetIpServerDescriptionAgentPrivate::setupSocket() { usedPort = port; timeoutHit = true; QKnxPrivate::clearSocket(&socket); Q_Q(QKnxNetIpServerDescriptionAgent); socket = new QUdpSocket(q); socket->setSocketOption(QUdpSocket::SocketOption::MulticastTtlOption, ttl); QObject::connect(socket, &QUdpSocket::stateChanged, [&](QUdpSocket::SocketState s) { Q_Q(QKnxNetIpServerDescriptionAgent); switch (s) { case QUdpSocket::BoundState: setAndEmitStateChanged(QKnxNetIpServerDescriptionAgent::State::Running); if (q->state() == QKnxNetIpServerDescriptionAgent::State::Running) { m_description = {}; usedPort = socket->localPort(); auto frame = QKnxNetIpDescriptionRequestProxy::builder() .setControlEndpoint(QKnxNetIpHpaiProxy::builder() .setHostAddress(nat ? QHostAddress::AnyIPv4 : socket->localAddress()) .setPort(nat ? quint16(0u) : usedPort).create()) .create(); const QKnxNetIpHpaiProxy hpai(m_server); socket->writeDatagram(frame.bytes().toByteArray(), hpai.hostAddress(), hpai.port()); setupAndStartReceiveTimer(); } break; default: break; } }); using overload = void (QUdpSocket::*)(QUdpSocket::SocketError); QObject::connect(socket, static_cast(&QUdpSocket::error), [&](QUdpSocket::SocketError) { setAndEmitErrorOccurred(QKnxNetIpServerDescriptionAgent::Error::Network, socket->errorString()); Q_Q(QKnxNetIpServerDescriptionAgent); q->stop(); }); QObject::connect(socket, &QUdpSocket::readyRead, [&]() { Q_Q(QKnxNetIpServerDescriptionAgent); while (socket->hasPendingDatagrams()) { if (q->state() != QKnxNetIpServerDescriptionAgent::State::Running) break; auto ba = socket->receiveDatagram().data(); QKnxByteArray data(ba.constData(), ba.size()); const auto header = QKnxNetIpFrameHeader::fromBytes(data, 0); if (!header.isValid() || header.serviceType() != QKnxNetIp::ServiceType::DescriptionResponse) continue; auto frame = QKnxNetIpFrame::fromBytes(data, 0); QKnxNetIpDescriptionResponseProxy response(frame); if (!response.isValid()) continue; setAndEmitServerDescriptionReceived({ m_server, response.deviceHardware(), #if QT_VERSION < QT_VERSION_CHECK(6, 0, 0) response.supportedFamilies() }); #else response.supportedFamilies(), {}, {} }); #endif } }); } namespace QKnxPrivate { static void clearTimer(QTimer **timer) { if (*timer) { (*timer)->stop(); (*timer)->disconnect(); (*timer)->deleteLater(); (*timer) = nullptr; } } } void QKnxNetIpServerDescriptionAgentPrivate::setupAndStartReceiveTimer() { Q_Q(QKnxNetIpServerDescriptionAgent); QKnxPrivate::clearTimer(&receiveTimer); if (timeout >= 0) { receiveTimer = new QTimer(q); receiveTimer->setSingleShot(true); receiveTimer->start(timeout); QObject::connect(receiveTimer, &QTimer::timeout, q, &QKnxNetIpServerDescriptionAgent::stop); } } void QKnxNetIpServerDescriptionAgentPrivate::setAndEmitStateChanged( QKnxNetIpServerDescriptionAgent::State newState) { state = newState; Q_Q(QKnxNetIpServerDescriptionAgent); emit q->stateChanged(newState); if (state == QKnxNetIpServerDescriptionAgent::State::Running) emit q->started(); else if (state == QKnxNetIpServerDescriptionAgent::State::NotRunning) emit q->finished(); } void QKnxNetIpServerDescriptionAgentPrivate::setAndEmitServerDescriptionReceived( const QKnxNetIpServerInfo &description) { timeoutHit = false; m_description = description; Q_Q(QKnxNetIpServerDescriptionAgent); emit q->descriptionReceived(description); } void QKnxNetIpServerDescriptionAgentPrivate::setAndEmitErrorOccurred( QKnxNetIpServerDescriptionAgent::Error newError, const QString &message) { error = newError; errorString = message; Q_Q(QKnxNetIpServerDescriptionAgent); emit q->errorOccurred(error, errorString); } // -- QKnxNetIpServerDescriptionAgent /*! Creates a KNXnet/IP server description agent with the parent \a parent. */ QKnxNetIpServerDescriptionAgent::QKnxNetIpServerDescriptionAgent(QObject *parent) : QKnxNetIpServerDescriptionAgent(QHostAddress(QHostAddress::AnyIPv4), 0u, parent) {} /*! Deletes a KNXnet/IP server description agent. */ QKnxNetIpServerDescriptionAgent::~QKnxNetIpServerDescriptionAgent() { stop(); } /*! Creates a KNXnet/IP server description agent with the host address \a localAddress and the parent \a parent. */ QKnxNetIpServerDescriptionAgent::QKnxNetIpServerDescriptionAgent(const QHostAddress &localAddress, QObject *parent) : QKnxNetIpServerDescriptionAgent(localAddress, 0u, parent) {} /*! Creates a KNXnet/IP server description agent with the host address \a localAddress, the port number \a port, and the parent \a parent. \note If the port number is already bound by a different process, description requests will fail. */ QKnxNetIpServerDescriptionAgent::QKnxNetIpServerDescriptionAgent(const QHostAddress &localAddress, quint16 port, QObject *parent) : QKnxNetIpServerDescriptionAgent(*new QKnxNetIpServerDescriptionAgentPrivate(localAddress, port), parent) {} /*! Returns the state of a KNXnet/IP server description agent. */ QKnxNetIpServerDescriptionAgent::State QKnxNetIpServerDescriptionAgent::state() const { Q_D(const QKnxNetIpServerDescriptionAgent); return d->state; } /*! Returns the type of an error that occurred when receiving the server description. */ QKnxNetIpServerDescriptionAgent::Error QKnxNetIpServerDescriptionAgent::error() const { Q_D(const QKnxNetIpServerDescriptionAgent); return d->error; } /*! Returns a human-readable string that describes an error. */ QString QKnxNetIpServerDescriptionAgent::errorString() const { Q_D(const QKnxNetIpServerDescriptionAgent); return d->errorString; } /*! Returns a description of a server. */ QKnxNetIpServerInfo QKnxNetIpServerDescriptionAgent::serverDescription() const { Q_D(const QKnxNetIpServerDescriptionAgent); return d->m_description; } /*! Returns the port number used by a description agent. */ quint16 QKnxNetIpServerDescriptionAgent::localPort() const { Q_D(const QKnxNetIpServerDescriptionAgent); if (d->state == QKnxNetIpServerDescriptionAgent::State::Running) return d->usedPort; return d->port; } /*! Sets the port number used by a description agent to \a port. \note If the port changes during a description request, the new port will not be used until the next run. */ void QKnxNetIpServerDescriptionAgent::setLocalPort(quint16 port) { Q_D(QKnxNetIpServerDescriptionAgent); if (d->state == QKnxNetIpServerDescriptionAgent::State::NotRunning) d->port = port; } /*! Returns the host address of a description agent. */ QHostAddress QKnxNetIpServerDescriptionAgent::localAddress() const { Q_D(const QKnxNetIpServerDescriptionAgent); return d->address; } /*! Sets the host address of a description agent to \a address. \note If the address changes during a description request, the new address will not be used until the next run. */ void QKnxNetIpServerDescriptionAgent::setLocalAddress(const QHostAddress &address) { Q_D(QKnxNetIpServerDescriptionAgent); if (d->state == QKnxNetIpServerDescriptionAgent::State::NotRunning) d->address = address; } /*! Returns the timeout value used by the description agent to wait for incoming search response messages. The default value is 3000 milliseconds. \sa setTimeout */ int QKnxNetIpServerDescriptionAgent::timeout() const { Q_D(const QKnxNetIpServerDescriptionAgent); return d->timeout; } /*! Sets the timeout for the description agent to \a msec. If \a msec is \c -1, the agent will not timeout and has to be terminated by calling the \l stop function. \sa timeout */ void QKnxNetIpServerDescriptionAgent::setTimeout(int msec) { Q_D(QKnxNetIpServerDescriptionAgent); d->timeout = msec; d->setupAndStartReceiveTimer(); } /*! Returns \c true if the server description agent uses network address translation (NAT). */ bool QKnxNetIpServerDescriptionAgent::natAware() const { Q_D(const QKnxNetIpServerDescriptionAgent); return d->nat; } /*! Sets whether the server description agent is using NAT to \a useNat. \note If the setting changes during a description request, it will not be used until the next run. */ void QKnxNetIpServerDescriptionAgent::setNatAware(bool useNat) { Q_D(QKnxNetIpServerDescriptionAgent); if (d->state == QKnxNetIpServerDescriptionAgent::State::NotRunning) d->nat = useNat; } /*! Returns the time to live (TTL) used for multicast search response messages. TTL is the maximum number of IP routers that may route the message. Each IP router that the message passes decrements the TTL by one. When the TTL has reached zero, the message is discarded. */ quint8 QKnxNetIpServerDescriptionAgent::multicastTtl() const { Q_D(const QKnxNetIpServerDescriptionAgent); return d->ttl; } /*! Sets the TTL used for multicasting to \a ttl. The value \c 1 means that the message does not leave the local network. */ void QKnxNetIpServerDescriptionAgent::setMulticastTtl(quint8 ttl) { Q_D(QKnxNetIpServerDescriptionAgent); d->ttl = ttl; if (d->socket) d->socket->setSocketOption(QUdpSocket::SocketOption::MulticastTtlOption, ttl); } /*! Starts the server description agent \a server. */ void QKnxNetIpServerDescriptionAgent::start(const QKnxNetIpHpai &server) { Q_D(QKnxNetIpServerDescriptionAgent); if (d->state != QKnxNetIpServerDescriptionAgent::State::NotRunning) return; auto isIPv4 = true; d->address.toIPv4Address(&isIPv4); if (isIPv4) { d->setAndEmitStateChanged(QKnxNetIpServerDescriptionAgent::State::Starting); d->setupSocket(); d->m_server = server; d->socket->bind(d->address, d->port); } else { d->setAndEmitErrorOccurred(Error::NotIPv4, tr("Only IPv4 local address supported.")); } } /*! Starts the server description agent \a server. */ void QKnxNetIpServerDescriptionAgent::start(const QKnxNetIpServerInfo &server) { start(server.endpoint()); } /*! Starts the server description agent at the host address \a address and port number \a port. */ void QKnxNetIpServerDescriptionAgent::start(const QHostAddress &address, quint16 port) { start(QKnxNetIpHpaiProxy::builder().setHostAddress(address).setPort(port).create()); } /*! Stops a server description agent. */ void QKnxNetIpServerDescriptionAgent::stop() { Q_D(QKnxNetIpServerDescriptionAgent); if (d->state == State::Stopping || d->state == State::NotRunning) return; d->setAndEmitStateChanged(QKnxNetIpServerDescriptionAgent::State::Stopping); d->socket->close(); QKnxPrivate::clearSocket(&(d->socket)); QKnxPrivate::clearTimer(&(d->receiveTimer)); if (d->timeoutHit) { emit d->setAndEmitErrorOccurred(Error::Timeout, tr("A timeout occurred while waiting for " "the description response.")); } d->setAndEmitStateChanged(QKnxNetIpServerDescriptionAgent::State::NotRunning); } QKnxNetIpServerDescriptionAgent::QKnxNetIpServerDescriptionAgent( QKnxNetIpServerDescriptionAgentPrivate &dd, QObject *parent) : QObject(dd, parent) {} QT_END_NAMESPACE