restructured to proper message allocation

This commit is contained in:
bernhard.trinnes 2020-07-30 10:47:01 +02:00 committed by Simon Stürz
parent f8df49845b
commit f0310205ad
7 changed files with 286 additions and 85 deletions

View File

@ -36,12 +36,25 @@ IntegrationPluginSma::IntegrationPluginSma()
}
void IntegrationPluginSma::discoverThings(ThingDiscoveryInfo *info)
{
if (info->thingClassId() == sunnyWebBoxThingClassId) {
info->finish(Thing::ThingErrorNoError);
}
}
void IntegrationPluginSma::setupThing(ThingSetupInfo *info)
{
Thing *thing = info->thing();
if (!m_udpSocket) {
m_udpSocket = new QUdpSocket(this);
if (!m_sunnyWebBoxCommunication) {
m_sunnyWebBoxCommunication = new SunnyWebBoxCommunication(this);
}
if (!m_refreshTimer) {
m_refreshTimer = hardwareManager()->pluginTimerManager()->registerTimer(10);
connect(m_refreshTimer, &PluginTimer::timeout, this, &IntegrationPluginSma::onRefreshTimer);
}
if (thing->thingClassId() == sunnyWebBoxThingClassId) {
@ -54,6 +67,10 @@ void IntegrationPluginSma::setupThing(ThingSetupInfo *info)
return;
}
}
SunnyWebBox *sunnyWebBox = new SunnyWebBox(m_sunnyWebBoxCommunication, QHostAddress(thing->paramValue(sunnyWebBoxThingHostParamTypeId).toString()), this);
m_sunnyWebBoxes.insert(thing, sunnyWebBox);
//TODO m_asyncSetup
return info->finish(Thing::ThingErrorNoError);
} else if (thing->thingClassId() == inverterThingClassId) {
Thing *parentThing = myThings().findById(thing->parentId());
@ -84,6 +101,26 @@ void IntegrationPluginSma::postSetupThing(Thing *thing)
return;
sunnyWebBox->getDevices();
} else if (thing->thingClassId() == inverterThingClassId) {
}
}
void IntegrationPluginSma::executeAction(ThingActionInfo *info)
{
Thing *thing = info->thing();
Action action = info->action();
if (thing->thingClassId() == sunnyWebBoxThingClassId) {
SunnyWebBox *sunnyWebBox = m_sunnyWebBoxes.value(thing);
if (!sunnyWebBox)
return;
if (action.actionTypeId() == sunnyWebBoxSearchDevicesActionTypeId) {
sunnyWebBox->getDevices();
} else {
//Unhandled actionTypeId
}
} else {
Q_ASSERT_X(false, "executeAction", QString("Unhandled thingClassId: %1").arg(thing->thingClassId().toString()).toUtf8());
}
}
@ -94,16 +131,58 @@ void IntegrationPluginSma::thingRemoved(Thing *thing)
}
if (myThings().filterByThingClassId(sunnyWebBoxThingClassId).isEmpty()) {
m_udpSocket->deleteLater();
m_udpSocket = nullptr;
m_sunnyWebBoxCommunication->deleteLater();
m_sunnyWebBoxCommunication = nullptr;
}
}
void IntegrationPluginSma::onRefreshTimer()
{
Q_FOREACH(SunnyWebBox *sunnyWebBox, m_sunnyWebBoxes) {
sunnyWebBox->getPlantOverview();
}
}
void IntegrationPluginSma::onPlantOverviewReceived(int messageId, SunnyWebBox::Overview overview)
{
Q_UNUSED(messageId)
Thing *thing = m_sunnyWebBoxes.key(static_cast<SunnyWebBox *>(sender()));
if (!thing)
return;
thing->setStateValue(sunnyWebBoxConnectedStateTypeId, true);
thing->setStateValue(sunnyWebBoxCurrentPowerStateTypeId, overview.power);
thing->setStateValue(sunnyWebBoxDayEnergyStateTypeId, overview.dailyYield);
thing->setStateValue(sunnyWebBoxTotalEnergyStateTypeId, overview.totalYield);
thing->setStateValue(sunnyWebBoxModeStateTypeId, overview.status);
if (!overview.error.isEmpty()){
qCDebug(dcSma()) << "Received error" << overview.error;
thing->setStateValue(sunnyWebBoxErrorStateTypeId, overview.error);
}
}
void IntegrationPluginSma::onDevicesReceived(int messageId, QList<SunnyWebBox::Device> devices)
{
Q_UNUSED(messageId)
Thing *thing = m_sunnyWebBoxes.key(static_cast<SunnyWebBox *>(sender()));
if (!thing)
return;
ThingDescriptors descriptors;
Q_FOREACH(SunnyWebBox::Device device, devices){
ThingDescriptor descriptor(inverterThingClassId, device.name, device.key ,thing->id());
descriptors.append(descriptor);
}
emit autoThingsAppeared(descriptors);
}
SunnyWebBox * IntegrationPluginSma::createSunnyWebBoxConnection(Thing *thing)
{
SunnyWebBox *sunnyWebBox = new SunnyWebBox(m_udpSocket, this);
SunnyWebBox *sunnyWebBox = new SunnyWebBox(m_sunnyWebBoxCommunication, QHostAddress(thing->paramValue(sunnyWebBoxThingHostParamTypeId).toString()), this);
m_sunnyWebBoxes.insert(thing, sunnyWebBox);
//connect();
connect(sunnyWebBox, &SunnyWebBox::plantOverviewReceived, this, &IntegrationPluginSma::onPlantOverviewReceived);
return sunnyWebBox;
}

View File

@ -34,6 +34,7 @@
#include "integrations/integrationplugin.h"
#include "plugintimer.h"
#include "sunnywebbox.h"
#include "sunnywebboxcommunication.h"
#include <QDebug>
#include <QHostAddress>
@ -48,18 +49,23 @@ class IntegrationPluginSma: public IntegrationPlugin {
public:
explicit IntegrationPluginSma();
void discoverThings(ThingDiscoveryInfo *info) override;
void setupThing(ThingSetupInfo *info) override;
void postSetupThing(Thing *thing) override;
void executeAction(ThingActionInfo *info) override;
void thingRemoved(Thing *thing) override;
private slots:
void onRefreshTimer();
void onPlantOverviewReceived(int messageId, SunnyWebBox::Overview overview);
void onDevicesReceived(int messageId, QList<SunnyWebBox::Device> devices);
private:
PluginTimer *m_refreshTimer = nullptr;
QHash<Thing *, SunnyWebBox *> m_sunnyWebBoxes;
QHash<Thing *, ThingSetupInfo *> m_asyncSetup;
QUdpSocket *m_udpSocket = nullptr;
SunnyWebBoxCommunication *m_sunnyWebBoxCommunication = nullptr;
SunnyWebBox * createSunnyWebBoxConnection(Thing *thing);
void setupChild(ThingSetupInfo *info, Thing *parentThing);

View File

@ -6,7 +6,9 @@ QT += \
SOURCES += \
integrationpluginsma.cpp \
sunnywebbox.cpp \
sunnywebboxcommunication.cpp \
HEADERS += \
integrationpluginsma.h \
sunnywebbox.h \
sunnywebboxcommunication.h \

View File

@ -34,98 +34,38 @@
#include "QJsonDocument"
#include "QJsonObject"
SunnyWebBox::SunnyWebBox(QUdpSocket *udpSocket, QObject *parrent) :
SunnyWebBox::SunnyWebBox(SunnyWebBoxCommunication *communication, const QHostAddress &hostAddress, QObject *parrent) :
QObject(parrent),
m_udpSocket(udpSocket)
m_hostAddresss(hostAddress),
m_communication(communication)
{
connect(m_udpSocket, &QUdpSocket::stateChanged, this, [this](QAbstractSocket::SocketState state) {
emit connectedChanged((state == QAbstractSocket::SocketState::ConnectedState));
});
connect(m_udpSocket, &QUdpSocket::readyRead, this, [this] {
//m_udpSocket->readDatagram(QByteArray())
qCDebug(dcSma()) << "Received datagram" << m_udpSocket->readAll();
});
//TODO connect communication with socket state;
connect(m_communication, &SunnyWebBoxCommunication::messageReceived, this, &SunnyWebBox::onMessageReceived);
}
int SunnyWebBox::getPlantOverview()
{
return sendMessage("GetPlantOverview");
return m_communication->sendMessage(m_hostAddresss, "GetPlantOverview");
}
int SunnyWebBox::getDevices()
{
return sendMessage("GetDevices");
return m_communication->sendMessage(m_hostAddresss, "GetDevices");
}
int SunnyWebBox::getProcessDataChannels(const QString &deviceId)
{
QJsonObject params;
params["device"] = deviceId;
return sendMessage("GetProcessDataChannels", params);
return m_communication->sendMessage(m_hostAddresss, "GetProcessDataChannels", params);
}
int SunnyWebBox::sendMessage(const QString &procedure)
void SunnyWebBox::onMessageReceived(const QHostAddress &address, int messageId, const QString &messageType, const QVariantMap &result)
{
int requestId = qrand();
QJsonDocument doc;
QJsonObject obj;
obj["version"] = "1.0";
obj["proc"] = procedure;
obj["id"] = requestId;
obj["format"] = "JSON";
m_udpSocket->writeDatagram(doc.toJson(), m_hostAddresss, m_port);
return requestId;
}
int SunnyWebBox::sendMessage(const QString &procedure, const QJsonObject &params)
{
int requestId = qrand();
QJsonDocument doc;
QJsonObject obj;
obj["version"] = "1.0";
obj["proc"] = procedure;
obj["id"] = requestId;
obj["format"] = "JSON";
if (!params.isEmpty()) {
obj.insert("params", params);
}
m_udpSocket->writeDatagram(doc.toJson(), m_hostAddresss, m_port);
return requestId;
}
void SunnyWebBox::onDatagramReceived(const QByteArray &data)
{
QJsonParseError error;
QJsonDocument doc = QJsonDocument::fromJson(data, &error);
if (error.error != QJsonParseError::NoError) {
qCWarning(dcSma()) << "Could not parse JSON" << error.errorString();
return;
}
if (!doc.isObject()) {
qCWarning(dcSma()) << "JSON is not an Object";
return;
}
QVariantMap map = doc.toVariant().toMap();
if (map["version"] != "1.0") {
qCWarning(dcSma()) << "API version not supported" << map["version"];
if (address != m_hostAddresss) {
return;
}
if (map.contains("proc") && map.contains("result")) {
QString requestType = map["proc"].toString();
int requestId = map["id"].toInt();
QVariantMap result = map.value("result").toMap();
emit messageResponseReceived(requestId, requestType, result);
} else {
qCWarning(dcSma()) << "Missing proc or result value";
}
}
void SunnyWebBox::parseMessageReponse(int messageId, const QString &messageType, const QVariantMap &result)
{
if (messageType == "GetPlantOverview") {
Overview overview;
QVariantList overviewList = result.value("overview").toList();

View File

@ -32,6 +32,7 @@
#define SUNNYWEBBOX_H
#include "integrations/thing.h"
#include "sunnywebboxcommunication.h"
#include <QObject>
#include <QHostAddress>
@ -56,7 +57,7 @@ public:
QList<Device> childrens;
};
explicit SunnyWebBox(QUdpSocket *udpSocket, QObject *parrent = 0);
explicit SunnyWebBox(SunnyWebBoxCommunication *communication, const QHostAddress &hostAddress, QObject *parrent = 0);
int getPlantOverview();
int getDevices();
@ -69,20 +70,17 @@ public:
QHostAddress hostAddress();
private:
int m_port = 34268;
QHostAddress m_hostAddresss;
QUdpSocket *m_udpSocket = nullptr;
int sendMessage(const QString &procedure);
int sendMessage(const QString &procedure, const QJsonObject &params);
QHostAddress m_hostAddresss;
SunnyWebBoxCommunication *m_communication = nullptr;
public slots:
void onDatagramReceived(const QByteArray &data);
void parseMessageReponse(int messageId, const QString &messageType, const QVariantMap &result);
void onMessageReceived(const QHostAddress &address, int messageId, const QString &messageType, const QVariantMap &result);
signals:
void connectedChanged(bool connected);
void messageResponseReceived(int messageId, const QString &messageType, const QVariantMap &result);
void plantOverviewReceived(int messageId, Overview overview);
void devicesReceived(int messageId, QList<Device> devices);
};

View File

@ -0,0 +1,118 @@
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
*
* 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 Lesser General Public License Usage
* Alternatively, this project may be redistributed and/or modified under the
* terms of the GNU Lesser General Public License as published by the Free
* Software Foundation; 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this project. If not, see <https://www.gnu.org/licenses/>.
*
* 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 "sunnywebboxcommunication.h"
#include "extern-plugininfo.h"
#include "QJsonDocument"
#include "QJsonObject"
SunnyWebBoxCommunication::SunnyWebBoxCommunication(QObject *parent) : QObject(parent)
{
m_udpSocket = new QUdpSocket(this);
m_udpSocket->bind(QHostAddress::LocalHost, m_port);
connect(m_udpSocket, &QUdpSocket::stateChanged, this, [this](QAbstractSocket::SocketState state) {
emit socketConnected(state == QAbstractSocket::SocketState::ConnectedState);
});
connect(m_udpSocket, &QUdpSocket::readyRead, this, [this] {
QHostAddress address;
quint16 port;
QByteArray data;
while (m_udpSocket->hasPendingDatagrams()) {
qCDebug(dcSma()) << "Received datagram";
int receivedBytes = m_udpSocket->readDatagram(data.data(), 1000, &address, &port);
if (receivedBytes == -1) {
qCWarning(dcSma()) << "Error reading pending datagram";
}
}
});
}
int SunnyWebBoxCommunication::sendMessage(const QHostAddress &address, const QString &procedure)
{
int requestId = qrand();
QJsonDocument doc;
QJsonObject obj;
obj["version"] = "1.0";
obj["proc"] = procedure;
obj["id"] = requestId;
obj["format"] = "JSON";
m_udpSocket->writeDatagram(doc.toJson(), address, m_port);
return requestId;
}
int SunnyWebBoxCommunication::sendMessage(const QHostAddress &address, const QString &procedure, const QJsonObject &params)
{
int requestId = qrand();
QJsonDocument doc;
QJsonObject obj;
obj["version"] = "1.0";
obj["proc"] = procedure;
obj["id"] = requestId;
obj["format"] = "JSON";
if (!params.isEmpty()) {
obj.insert("params", params);
}
m_udpSocket->writeDatagram(doc.toJson(), address, m_port);
return requestId;
}
void SunnyWebBoxCommunication::datagramReceived(const QHostAddress &address, const QByteArray &data)
{
QJsonParseError error;
QJsonDocument doc = QJsonDocument::fromJson(data, &error);
if (error.error != QJsonParseError::NoError) {
qCWarning(dcSma()) << "Could not parse JSON" << error.errorString();
return;
}
if (!doc.isObject()) {
qCWarning(dcSma()) << "JSON is not an Object";
return;
}
QVariantMap map = doc.toVariant().toMap();
if (map["version"] != "1.0") {
qCWarning(dcSma()) << "API version not supported" << map["version"];
return;
}
if (map.contains("proc") && map.contains("result")) {
QString requestType = map["proc"].toString();
int requestId = map["id"].toInt();
QVariantMap result = map.value("result").toMap();
emit messageReceived(address, requestId, requestType, result);
} else {
qCWarning(dcSma()) << "Missing proc or result value";
}
}

View File

@ -0,0 +1,58 @@
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
*
* 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 Lesser General Public License Usage
* Alternatively, this project may be redistributed and/or modified under the
* terms of the GNU Lesser General Public License as published by the Free
* Software Foundation; 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this project. If not, see <https://www.gnu.org/licenses/>.
*
* 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
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
#ifndef SUNNYWEBBOXCOMMUNICATION_H
#define SUNNYWEBBOXCOMMUNICATION_H
#include <QObject>
#include <QUdpSocket>
class SunnyWebBoxCommunication : public QObject
{
Q_OBJECT
public:
explicit SunnyWebBoxCommunication(QObject *parent = nullptr);
int sendMessage(const QHostAddress &address, const QString &procedure);
int sendMessage(const QHostAddress &address, const QString &procedure, const QJsonObject &params);
private:
int m_port = 34268;
QUdpSocket *m_udpSocket;
void datagramReceived(const QHostAddress &address, const QByteArray &data);
signals:
void socketConnected(bool connected);
void messageReceived(const QHostAddress &address, int messageId, const QString &messageType, const QVariantMap &result);
};
#endif // SUNNYWEBBOXCOMMUNICATION_H