powersync-plugins-modbus/sma/sunnywebbox/sunnywebbox.cpp

347 lines
14 KiB
C++

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
*
* 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 "sunnywebbox.h"
#include "extern-plugininfo.h"
#include <QJsonDocument>
#include <QJsonObject>
#include <QJsonArray>
SunnyWebBox::SunnyWebBox(NetworkAccessManager *networkAccessManager, const QHostAddress &hostAddress, QObject *parrent) :
QObject(parrent),
m_networkManager(networkAccessManager),
m_hostAddresss(hostAddress)
{
qCDebug(dcSma()) << "SunnyWebBox: Creating Sunny Web Box connection";
}
SunnyWebBox::~SunnyWebBox()
{
qCDebug(dcSma()) << "SunnyWebBox: Deleting Sunny Web Box connection";
}
QString SunnyWebBox::getPlantOverview()
{
QDateTime currentDateTime = QDateTime::currentDateTime();
if (!m_lastRequest.isNull()) {
// According to the documentation, the interval between 2 requests should not be less than 30 seconds.
if (QDateTime::currentMSecsSinceEpoch() - m_lastRequest.toMSecsSinceEpoch() < 30000) {
return QString();
}
}
m_lastRequest = currentDateTime;
return sendMessage(m_hostAddresss, "GetPlantOverview");
}
QString SunnyWebBox::getDevices()
{
return sendMessage(m_hostAddresss, "GetDevices");
}
QString SunnyWebBox::getProcessDataChannels(const QString &deviceId)
{
QJsonObject params;
params["device"] = deviceId;
return sendMessage(m_hostAddresss, "GetProcessDataChannels", params);
}
QString SunnyWebBox::getProcessData(const QStringList &deviceKeys)
{
QJsonObject paramsObj;
QJsonArray devicesArray;
foreach (const QString &key, deviceKeys) {
QJsonObject deviceObj;
deviceObj["key"] = key;
devicesArray.append(deviceObj);
}
paramsObj["devices"] = devicesArray;
return sendMessage(m_hostAddresss, "GetProcessData", paramsObj);
}
QString SunnyWebBox::getParameterChannels(const QString &deviceKey)
{
QJsonObject paramsObj;
QJsonArray devicesArray;
QJsonObject deviceObj;
deviceObj["key"] = deviceKey;
devicesArray.append(deviceObj);
paramsObj["devices"] = devicesArray;
return sendMessage(m_hostAddresss, "GetParameterChannels", paramsObj);
}
QString SunnyWebBox::getParameters(const QStringList &deviceKeys)
{
QJsonObject paramsObj;
QJsonArray devicesArray;
foreach (const QString &key, deviceKeys) {
QJsonObject deviceObj;
deviceObj["key"] = key;
devicesArray.append(deviceObj);
}
paramsObj["devices"] = devicesArray;
return sendMessage(m_hostAddresss, "GetParameter", paramsObj);
}
QString SunnyWebBox::setParameters(const QString &deviceKey, const QHash<QString, QVariant> &channels)
{
QJsonObject paramsObj;
QJsonArray devicesArray;
QJsonObject deviceObj;
deviceObj["key"] = deviceKey;
QJsonArray channelsArray;
foreach (const QString &key, channels.keys()) {
QJsonObject channelObj;
channelObj["meta"] = key;
channelObj["value"] = channels.value(key).toString();
channelsArray.append(channelObj);
}
deviceObj["channels"] = channelsArray;
devicesArray.append(deviceObj);
paramsObj["devices"] = devicesArray;
return sendMessage(m_hostAddresss, "SetParameter", paramsObj);
}
QHostAddress SunnyWebBox::hostAddress() const
{
return m_hostAddresss;
}
void SunnyWebBox::setHostAddress(const QHostAddress &address)
{
qCDebug(dcSma()) << "SunnyWebBox: Setting host address to" << address.toString();
m_hostAddresss = address;
}
QNetworkReply *SunnyWebBox::sendRequest(const QHostAddress &address, const QString &procedure, const QJsonObject &params, const QString &requestId)
{
qCDebug(dcSma()) << "SunnyWebBox: Send message to" << address.toString() << "Procedure:" << procedure << "Params:" << params;
QString finalRequestId = requestId;
if (finalRequestId.isEmpty())
finalRequestId = generateRequestId();
QJsonDocument doc;
QJsonObject obj;
obj["format"] = "JSON";
obj["id"] = requestId;
obj["proc"] = procedure;
obj["version"] = "1.0";
if (!params.isEmpty()) {
obj.insert("params", params);
}
doc.setObject(obj);
QUrl url;
url.setScheme("http");
url.setHost(address.toString());
url.setPath("/rpc");
url.setPort(80);
QNetworkRequest request(url);
request.setHeader(QNetworkRequest::KnownHeaders::ContentTypeHeader, "application/json");
QByteArray data = doc.toJson(QJsonDocument::JsonFormat::Compact);
data.prepend("RPC=");
return m_networkManager->post(request, data);
}
QString SunnyWebBox::generateRequestId()
{
return QUuid::createUuid().toString().remove('{').remove('-').left(14);
}
void SunnyWebBox::parseMessage(const QString &messageId, const QString &messageType, const QVariantMap &result)
{
if (messageType == "GetPlantOverview") {
Overview overview;
QVariantList overviewList = result.value("overview").toList();
qCDebug(dcSma()) << "SunnyWebBox: GetPlantOverview";
foreach (const QVariant &value, overviewList) {
QVariantMap map = value.toMap();
if (map["meta"].toString() == "GriPwr") {
overview.power = map["value"].toString().toDouble();
QString unit = map["unit"].toString();
qCDebug(dcSma()) << "SunnyWebBox: - Power" << overview.power << unit;
} else if (map["meta"].toString() == "GriEgyTdy") {
overview.dailyYield = map["value"].toString().toDouble();
QString unit = map["unit"].toString();
qCDebug(dcSma()) << "SunnyWebBox: - Daily yield" << overview.dailyYield << unit;
} else if (map["meta"].toString() == "GriEgyTot") {
overview.totalYield = map["value"].toString().toDouble();
QString unit = map["unit"].toString();
qCDebug(dcSma()) << "SunnyWebBox: - Total yield" << overview.totalYield << unit;
} else if (map["meta"].toString() == "OpStt") {
overview.status = map["value"].toString();
qCDebug(dcSma()) << "SunnyWebBox: - Status" << overview.status;
} else if (map["meta"].toString() == "Msg") {
overview.error = map["value"].toString();
qCDebug(dcSma()) << "SunnyWebBox: - Error" << overview.error;
}
}
emit plantOverviewReceived(messageId, overview);
} else if (messageType == "GetDevices") {
QList<Device> devices;
QVariantList deviceList = result.value("devices").toList();
qCDebug(dcSma()) << "SunnyWebBox: GetDevices" << result.value("totalDevicesReturned").toInt();
foreach (const QVariant &value, deviceList) {
Device device;
QVariantMap map = value.toMap();
device.name = map["name"].toString();
qCDebug(dcSma()) << "SunnyWebBox: - Name" << device.name;
device.key = map["key"].toString();
qCDebug(dcSma()) << "SunnyWebBox: - Key" << device.key;
QVariantList childrenList = map["children"].toList();
foreach (const QVariant &childValue, childrenList) {
Device child;
QVariantMap childMap = childValue.toMap();
device.name = childMap["name"].toString();
device.key = childMap["key"].toString();
device.childrens.append(child);
}
devices.append(device);
}
if (!devices.isEmpty()) {
emit devicesReceived(messageId, devices);
}
} else if (messageType == "GetProcessDataChannels" ||
messageType == "GetProDataChannels") {
foreach (const QString &deviceKey, result.keys()) {
QStringList processDataChannels = result.value(deviceKey).toStringList();
if (!processDataChannels.isEmpty())
emit processDataChannelsReceived(messageId, deviceKey, processDataChannels);
}
} else if (messageType == "GetProcessData") {
QVariantList devicesList = result.value("devices").toList();
qCDebug(dcSma()) << "SunnyWebBox: GetProcessData response received";
foreach (const QVariant &value, devicesList) {
QString key = value.toMap().value("key").toString();
QVariantList channelsList = value.toMap().value("channels").toList();
QHash<QString, QVariant> channels;
foreach (const QVariant &channel, channelsList) {
channels.insert(channel.toMap().value("meta").toString(), channel.toMap().value("value"));
}
emit processDataReceived(messageId, key, channels);
}
} else if (messageType == "GetParameterChannels") {
foreach (const QString &deviceKey, result.keys()) {
QStringList parameterChannels = result.value(deviceKey).toStringList();
if (!parameterChannels.isEmpty())
emit parameterChannelsReceived(messageId, deviceKey, parameterChannels);
}
} else if (messageType == "GetParameter"|| messageType == "SetParameter") {
QVariantList devicesList = result.value("devices").toList();
foreach (const QVariant &value, devicesList) {
QString key = value.toMap().value("key").toString();
QVariantList channelsList = value.toMap().value("channels").toList();
QList<Parameter> parameters;
foreach (const QVariant &channel, channelsList) {
Parameter parameter;
parameter.meta = channel.toMap().value("meta").toString();
parameter.name = channel.toMap().value("name").toString();
parameter.unit = channel.toMap().value("unit").toString();
parameter.min = channel.toMap().value("min").toDouble();
parameter.max = channel.toMap().value("max").toDouble();
parameter.value = channel.toMap().value("value").toDouble();
parameters.append(parameter);
}
emit parametersReceived(messageId, key, parameters);
}
} else {
qCWarning(dcSma()) << "SunnyWebBox: Unknown message type" << messageType;
}
}
void SunnyWebBox::setConnectionStatus(bool connected)
{
if (m_connected != connected) {
qCDebug(dcSma()) << "SunnyWebBox: Connection status changed" << connected;
m_connected = connected;
emit connectedChanged(m_connected);
}
}
QString SunnyWebBox::sendMessage(const QHostAddress &address, const QString &procedure)
{
return sendMessage(address, procedure, QJsonObject());
}
QString SunnyWebBox::sendMessage(const QHostAddress &address, const QString &procedure, const QJsonObject &params)
{
QString requestId = generateRequestId();
QNetworkReply *reply = sendRequest(m_hostAddresss, procedure, params, requestId);
connect(reply, &QNetworkReply::finished, reply, &QNetworkReply::deleteLater);
connect(reply, &QNetworkReply::finished, this, [this, address, requestId, reply]{
if (reply->error() != QNetworkReply::NoError) {
setConnectionStatus(false);
return;
}
setConnectionStatus(true);
QByteArray data = reply->readAll();
qCDebug(dcSma()) << "SunnyWebBox: Received reply" << data;
QJsonParseError error;
QJsonDocument doc = QJsonDocument::fromJson(data, &error);
if (error.error != QJsonParseError::NoError) {
qCWarning(dcSma()) << "SunnyWebBox: Could not parse JSON" << error.errorString();
return;
}
if (!doc.isObject()) {
qCWarning(dcSma()) << "SunnyWebBox: JSON is not an Object";
return;
}
QVariantMap map = doc.toVariant().toMap();
if (map["version"] != "1.0") {
qCWarning(dcSma()) << "SunnyWebBox: API version not supported" << map["version"];
return;
}
if (map.contains("proc") && map.contains("result")) {
QString requestType = map["proc"].toString();
QString requestId = map["id"].toString();
QVariantMap result = map.value("result").toMap();
parseMessage(requestId, requestType, result);
} else if (map.contains("proc") && map.contains("error")) {
} else {
qCWarning(dcSma()) << "SunnyWebBox: Missing proc or result value";
}
});
return requestId;
}