|
@ -7,6 +7,7 @@ |
|
|
*/ |
|
|
*/ |
|
|
|
|
|
|
|
|
#include <QLoggingCategory>
|
|
|
#include <QLoggingCategory>
|
|
|
|
|
|
#include <QtMqtt/QtMqtt>
|
|
|
#include <QJsonDocument>
|
|
|
#include <QJsonDocument>
|
|
|
#include <QJsonObject>
|
|
|
#include <QJsonObject>
|
|
|
#include <QJsonValue>
|
|
|
#include <QJsonValue>
|
|
@ -14,57 +15,76 @@ |
|
|
|
|
|
|
|
|
#include "mqttclient.h"
|
|
|
#include "mqttclient.h"
|
|
|
#include "settings.h"
|
|
|
#include "settings.h"
|
|
|
#include "qmqtt.h"
|
|
|
|
|
|
|
|
|
|
|
|
MQTTClient::MQTTClient(const QHostAddress& host, const quint16 port, |
|
|
|
|
|
QObject* parent) : QMQTT::Client(host, port, parent) |
|
|
|
|
|
|
|
|
MQTTClient::MQTTClient(const QString& host, int port, QObject* parent) |
|
|
|
|
|
: QMqttClient(parent) |
|
|
{ |
|
|
{ |
|
|
|
|
|
this->setHostname(host); |
|
|
|
|
|
this->setPort(port); |
|
|
|
|
|
|
|
|
connect(this, SIGNAL(connected(void)), this, SLOT(onConnected(void))); |
|
|
connect(this, SIGNAL(connected(void)), this, SLOT(onConnected(void))); |
|
|
connect(this, SIGNAL(subscribed(const QString&)), this, SLOT(onSubscribed(const QString&))); |
|
|
|
|
|
connect(this, SIGNAL(received(const QMQTT::Message&)), this, SLOT(onReceived(const QMQTT::Message&))); |
|
|
|
|
|
connect(this, SIGNAL(published(const QMQTT::Message&, quint16)), this, SLOT(onPublished(const QMQTT::Message&, quint16))); |
|
|
|
|
|
connect(this, SIGNAL(error(const QMQTT::ClientError)), this, SLOT(onError(const QMQTT::ClientError))); |
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
* 2 more signals: |
|
|
|
|
|
* void unsubscribed(const QString& topic); |
|
|
|
|
|
* void disconnected(void); |
|
|
|
|
|
* void pingresp(); |
|
|
|
|
|
*/ |
|
|
|
|
|
|
|
|
connect(this, SIGNAL(disconnected(void)), this, SLOT(onDisconnected(void))); |
|
|
|
|
|
connect(this, SIGNAL(messageReceived(const QByteArray &, const QMqttTopicName &)), |
|
|
|
|
|
this, SLOT(onReceived(const QByteArray &, const QMqttTopicName &))); |
|
|
|
|
|
connect(this, SIGNAL(messageSent(qint32)), this, SLOT(onPublished(qint32))); |
|
|
|
|
|
connect(this, SIGNAL(errorChanged(ClientError)), |
|
|
|
|
|
this, SLOT(onError(ClientError))); |
|
|
|
|
|
connect(this, SIGNAL(stateChanged(ClientState)), |
|
|
|
|
|
this, SLOT(onStateChanged(ClientState))); |
|
|
|
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
MQTTClient::~MQTTClient() { |
|
|
MQTTClient::~MQTTClient() { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void MQTTClient::onError(const QMQTT::ClientError error) |
|
|
|
|
|
|
|
|
void MQTTClient::onStateChanged(ClientState state) |
|
|
{ |
|
|
{ |
|
|
qDebug() << "error" << error; |
|
|
|
|
|
|
|
|
qDebug() << "mqtt state " << state; |
|
|
|
|
|
switch (state) { |
|
|
|
|
|
case Disconnected: break; |
|
|
|
|
|
case Connecting: break; |
|
|
|
|
|
case Connected: break; |
|
|
|
|
|
default: break; |
|
|
|
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void MQTTClient::onConnected() |
|
|
|
|
|
|
|
|
void MQTTClient::onError(ClientError error) |
|
|
{ |
|
|
{ |
|
|
qDebug() << "connected"; |
|
|
|
|
|
this->subscribe("sensors/#", 1); |
|
|
|
|
|
|
|
|
qDebug() << "mqtt error " << error; |
|
|
|
|
|
/* TODO */ |
|
|
|
|
|
switch (error) { |
|
|
|
|
|
case NoError: break; |
|
|
|
|
|
case InvalidProtocolVersion: break; |
|
|
|
|
|
case IdRejected: break; |
|
|
|
|
|
case ServerUnavailable: break; |
|
|
|
|
|
case BadUsernameOrPassword: break; |
|
|
|
|
|
case NotAuthorized: break; |
|
|
|
|
|
case TransportInvalid: break; |
|
|
|
|
|
case ProtocolViolation: break; |
|
|
|
|
|
case UnknownError: break; |
|
|
|
|
|
default: break; |
|
|
|
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void MQTTClient::onSubscribed(const QString& topic) |
|
|
|
|
|
|
|
|
void MQTTClient::onConnected() |
|
|
{ |
|
|
{ |
|
|
qDebug() << "subscribed " << topic; |
|
|
|
|
|
|
|
|
qDebug() << "connected"; |
|
|
|
|
|
this->subscribe(QString("sensors/#"), 1); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void MQTTClient::onReceived(const QMQTT::Message& message) |
|
|
|
|
|
|
|
|
void MQTTClient::onReceived(const QByteArray &message, const QMqttTopicName &topic) |
|
|
{ |
|
|
{ |
|
|
Settings *s = Settings::getInstance(); |
|
|
Settings *s = Settings::getInstance(); |
|
|
QJsonDocument sensorData; |
|
|
QJsonDocument sensorData; |
|
|
QJsonValue val; |
|
|
QJsonValue val; |
|
|
|
|
|
|
|
|
qDebug() << "publish received: \"" << QString::fromUtf8(message.payload()) |
|
|
|
|
|
<< "\"" << " from: " << message.topic(); |
|
|
|
|
|
|
|
|
qDebug() << "publish received: \"" << QString::fromUtf8(message) |
|
|
|
|
|
<< "\"" << " from: " << topic.name(); |
|
|
|
|
|
|
|
|
for (int i = 0; i < s->nbZones(); ++i) { |
|
|
for (int i = 0; i < s->nbZones(); ++i) { |
|
|
if (s->m_rooms.at(i).sensor_topic == message.topic()) { |
|
|
|
|
|
|
|
|
if (s->m_rooms.at(i).sensor_topic == topic.name()) { |
|
|
qDebug() << "this is for us !"; |
|
|
qDebug() << "this is for us !"; |
|
|
sensorData = QJsonDocument::fromJson(message.payload()); |
|
|
|
|
|
|
|
|
sensorData = QJsonDocument::fromJson(message); |
|
|
if (!sensorData.isObject()) { |
|
|
if (!sensorData.isObject()) { |
|
|
qWarning() << "malformed JSON data"; |
|
|
qWarning() << "malformed JSON data"; |
|
|
goto out; |
|
|
goto out; |
|
@ -91,31 +111,34 @@ void MQTTClient::onReceived(const QMQTT::Message& message) |
|
|
break; |
|
|
break; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
if (s->m_rooms.at(i).availability_topic == message.topic()) { |
|
|
|
|
|
|
|
|
if (s->m_rooms.at(i).availability_topic == topic.name()) { |
|
|
|
|
|
|
|
|
emit new_availability(i, message.payload() == QString("online").toUtf8()); |
|
|
|
|
|
|
|
|
emit new_availability(i, message == QString("online").toUtf8()); |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
out: |
|
|
out: |
|
|
return; |
|
|
return; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void MQTTClient::onPublished(const QMQTT::Message& message, quint16 msgid) |
|
|
|
|
|
|
|
|
void MQTTClient::onPublished(qint32 msgid) |
|
|
{ |
|
|
{ |
|
|
qDebug() << "published" << msgid << " topic: " << message.topic() << "payload" << QString::fromUtf8(message.payload()); |
|
|
|
|
|
|
|
|
qDebug() << "published id " << msgid; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void MQTTClient::onDisconnected(void) |
|
|
void MQTTClient::onDisconnected(void) |
|
|
{ |
|
|
{ |
|
|
|
|
|
/* TODO */ |
|
|
qDebug() << "disconnected"; |
|
|
qDebug() << "disconnected"; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void MQTTClient::publish_msg(const QString& topic, const QString& payload) |
|
|
void MQTTClient::publish_msg(const QString& topic, const QString& payload) |
|
|
{ |
|
|
{ |
|
|
// id is automatically incremented if ==0 and QoS>0
|
|
|
|
|
|
// DUP: if a message is resent, it should have the DUP flag
|
|
|
|
|
|
QMQTT::Message message(0, topic, payload.toUtf8(), 1, true, false); |
|
|
|
|
|
publish(message); |
|
|
|
|
|
|
|
|
qint32 msgid; |
|
|
|
|
|
|
|
|
|
|
|
msgid = this->publish(QMqttTopicName(topic), payload.toUtf8(), 1, true); |
|
|
|
|
|
if (msgid == -1) { |
|
|
|
|
|
qDebug() << "fail to publish message " << payload << " on " << topic; |
|
|
|
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void MQTTClient::allHeatersOn(bool on) { |
|
|
void MQTTClient::allHeatersOn(bool on) { |
|
|