/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Copyright (C) 2018 Michael Zanetti * * * * This file is part of nymea. * * * * nymea is free software: you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation, version 2 of the License. * * * * nymea 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 nymea. If not, see . * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ #include "deviceplugintasmota.h" #include "plugininfo.h" #include #include #include #include #include "hardwaremanager.h" #include "network/networkaccessmanager.h" #include "network/mqtt/mqttprovider.h" #include "network/mqtt/mqttchannel.h" DevicePluginTasmota::DevicePluginTasmota() { // Helper maps for parent devices (aka sonoff_*) m_ipAddressParamTypeMap[sonoff_basicDeviceClassId] = sonoff_basicDeviceIpAddressParamTypeId; m_ipAddressParamTypeMap[sonoff_dualDeviceClassId] = sonoff_dualDeviceIpAddressParamTypeId; m_ipAddressParamTypeMap[sonoff_quadDeviceClassId] = sonoff_quadDeviceIpAddressParamTypeId; m_attachedDeviceParamTypeIdMap[sonoff_basicDeviceClassId] << sonoff_basicDeviceAttachedDeviceCH1ParamTypeId; m_attachedDeviceParamTypeIdMap[sonoff_dualDeviceClassId] << sonoff_dualDeviceAttachedDeviceCH1ParamTypeId << sonoff_dualDeviceAttachedDeviceCH2ParamTypeId; m_attachedDeviceParamTypeIdMap[sonoff_quadDeviceClassId] << sonoff_quadDeviceAttachedDeviceCH1ParamTypeId << sonoff_quadDeviceAttachedDeviceCH2ParamTypeId << sonoff_quadDeviceAttachedDeviceCH3ParamTypeId << sonoff_quadDeviceAttachedDeviceCH4ParamTypeId; // Helper maps for virtual childs (aka tasmota*) m_channelParamTypeMap[tasmotaSwitchDeviceClassId] = tasmotaSwitchDeviceChannelNameParamTypeId; m_channelParamTypeMap[tasmotaLightDeviceClassId] = tasmotaLightDeviceChannelNameParamTypeId; m_powerStateTypeMap[tasmotaSwitchDeviceClassId] = tasmotaSwitchPowerStateTypeId; m_powerStateTypeMap[tasmotaLightDeviceClassId] = tasmotaLightPowerStateTypeId; // Helper maps for all devices m_connectedStateTypeMap[sonoff_basicDeviceClassId] = sonoff_basicConnectedStateTypeId; m_connectedStateTypeMap[sonoff_dualDeviceClassId] = sonoff_dualConnectedStateTypeId; m_connectedStateTypeMap[sonoff_quadDeviceClassId] = sonoff_quadConnectedStateTypeId; m_connectedStateTypeMap[tasmotaSwitchDeviceClassId] = tasmotaSwitchConnectedStateTypeId; m_connectedStateTypeMap[tasmotaLightDeviceClassId] = tasmotaLightConnectedStateTypeId; } DevicePluginTasmota::~DevicePluginTasmota() { } void DevicePluginTasmota::init() { } DeviceManager::DeviceSetupStatus DevicePluginTasmota::setupDevice(Device *device) { if (m_ipAddressParamTypeMap.contains(device->deviceClassId())) { ParamTypeId ipAddressParamTypeId = m_ipAddressParamTypeMap.value(device->deviceClassId()); QHostAddress deviceAddress = QHostAddress(device->paramValue(ipAddressParamTypeId).toString()); if (deviceAddress.isNull()) { qCWarning(dcTasmota) << "Not a valid IP address given for IP address parameter"; return DeviceManager::DeviceSetupStatusFailure; } MqttChannel *channel = hardwareManager()->mqttProvider()->createChannel(device->id(), deviceAddress); Q_UNUSED(device) QUrl url("http://10.10.10.90/sv"); QUrlQuery query; query.addQueryItem("w", "2%2C1"); query.addQueryItem("mh", channel->serverAddress().toString()); query.addQueryItem("ml", QString::number(channel->serverPort())); query.addQueryItem("mc", channel->clientId()); query.addQueryItem("mu", channel->username()); query.addQueryItem("mp", channel->password()); query.addQueryItem("mt", "sonoff"); query.addQueryItem("mf", channel->topicPrefix() + "/%topic%/"); url.setQuery(query); QNetworkRequest request(url); QNetworkReply *reply = hardwareManager()->networkManager()->get(request); connect(reply, &QNetworkReply::finished, this, [this, device, channel, reply](){ reply->deleteLater(); if (reply->error() != QNetworkReply::NoError) { qCDebug(dcTasmota) << "Sonoff device setup call failed:" << reply->error() << reply->errorString() << reply->readAll(); hardwareManager()->mqttProvider()->releaseChannel(channel); emit deviceSetupFinished(device, DeviceManager::DeviceSetupStatusFailure); return; } m_mqttChannels.insert(device, channel); connect(channel, &MqttChannel::clientConnected, this, &DevicePluginTasmota::onClientConnected); connect(channel, &MqttChannel::clientDisconnected, this, &DevicePluginTasmota::onClientDisconnected); connect(channel, &MqttChannel::publishReceived, this, &DevicePluginTasmota::onPublishReceived); qCDebug(dcTasmota) << "Sonoff setup complete"; emit deviceSetupFinished(device, DeviceManager::DeviceSetupStatusSuccess); foreach (Device *child, myDevices()) { if (child->parentId() == device->id()) { // Already have child devices... We're done here return; } } qCDebug(dcTasmota) << "Adding Tasmota Switch devices"; QList deviceDescriptors; for (int i = 0; i < m_attachedDeviceParamTypeIdMap.value(device->deviceClassId()).count(); i++) { DeviceDescriptor descriptor(tasmotaSwitchDeviceClassId, device->name() + " CH" + QString::number(i+1), QString(), device->id()); if (m_attachedDeviceParamTypeIdMap.value(device->deviceClassId()).count() == 1) { descriptor.setParams(ParamList() << Param(tasmotaSwitchDeviceChannelNameParamTypeId, "POWER")); } else { descriptor.setParams(ParamList() << Param(tasmotaSwitchDeviceChannelNameParamTypeId, "POWER" + QString::number(i+1))); } deviceDescriptors << descriptor; } emit autoDevicesAppeared(tasmotaSwitchDeviceClassId, deviceDescriptors); qCDebug(dcTasmota) << "Adding Tasmota connected devices"; deviceDescriptors.clear(); for (int i = 0; i < m_attachedDeviceParamTypeIdMap.value(device->deviceClassId()).count(); i++) { ParamTypeId attachedDeviceParamTypeId = m_attachedDeviceParamTypeIdMap.value(device->deviceClassId()).at(i); if (device->paramValue(attachedDeviceParamTypeId).toString() == "Light") { DeviceDescriptor descriptor1(tasmotaLightDeviceClassId, device->name() + " CH" + QString::number(i+1), QString(), device->id()); if (m_attachedDeviceParamTypeIdMap.value(device->deviceClassId()).count() == 1) { descriptor1.setParams(ParamList() << Param(tasmotaLightDeviceChannelNameParamTypeId, "POWER")); } else { descriptor1.setParams(ParamList() << Param(tasmotaLightDeviceChannelNameParamTypeId, "POWER" + QString::number(i+1))); } deviceDescriptors << descriptor1; } } if (!deviceDescriptors.isEmpty()) { emit autoDevicesAppeared(tasmotaLightDeviceClassId, deviceDescriptors); } }); return DeviceManager::DeviceSetupStatusAsync; } if (m_connectedStateTypeMap.contains(device->deviceClassId())) { Device* parentDevice = myDevices().findById(device->parentId()); StateTypeId connectedStateTypeId = m_connectedStateTypeMap.value(device->deviceClassId()); device->setStateValue(m_connectedStateTypeMap.value(device->deviceClassId()), parentDevice->stateValue(connectedStateTypeId)); return DeviceManager::DeviceSetupStatusSuccess; } qCWarning(dcTasmota) << "Unhandled DeviceClass in setupDevice" << device->deviceClassId(); return DeviceManager::DeviceSetupStatusFailure; } void DevicePluginTasmota::deviceRemoved(Device *device) { qCDebug(dcTasmota) << "Device removed" << device->name(); if (m_mqttChannels.contains(device)) { qCDebug(dcTasmota) << "Releasing MQTT channel"; MqttChannel* channel = m_mqttChannels.take(device); hardwareManager()->mqttProvider()->releaseChannel(channel); } } DeviceManager::DeviceError DevicePluginTasmota::executeAction(Device *device, const Action &action) { if (m_powerStateTypeMap.contains(device->deviceClassId())) { Device *parentDev = myDevices().findById(device->parentId()); MqttChannel *channel = m_mqttChannels.value(parentDev); ParamTypeId channelParamTypeId = m_channelParamTypeMap.value(device->deviceClassId()); ParamTypeId powerActionParamTypeId = ParamTypeId(m_powerStateTypeMap.value(device->deviceClassId()).toString()); qCDebug(dcTasmota) << "Publishing:" << channel->topicPrefix() + "/sonoff/cmnd/" + device->paramValue(channelParamTypeId).toString() << (action.param(powerActionParamTypeId).value().toBool() ? "ON" : "OFF"); channel->publish(channel->topicPrefix() + "/sonoff/cmnd/" + device->paramValue(channelParamTypeId).toString().toLower(), action.param(powerActionParamTypeId).value().toBool() ? "ON" : "OFF"); } return DeviceManager::DeviceErrorActionTypeNotFound; } void DevicePluginTasmota::onClientConnected(MqttChannel *channel) { qCDebug(dcTasmota) << "Sonoff device connected!"; Device *dev = m_mqttChannels.key(channel); dev->setStateValue(m_connectedStateTypeMap.value(dev->deviceClassId()), true); foreach (Device *child, myDevices()) { if (child->parentId() == dev->id()) { child->setStateValue(m_connectedStateTypeMap.value(child->deviceClassId()), true); } } } void DevicePluginTasmota::onClientDisconnected(MqttChannel *channel) { qCDebug(dcTasmota) << "Sonoff device disconnected!"; Device *dev = m_mqttChannels.key(channel); dev->setStateValue(m_connectedStateTypeMap.value(dev->deviceClassId()), false); foreach (Device *child, myDevices()) { if (child->parentId() == dev->id()) { child->setStateValue(m_connectedStateTypeMap.value(dev->deviceClassId()), false); } } } void DevicePluginTasmota::onPublishReceived(MqttChannel *channel, const QString &topic, const QByteArray &payload) { qCDebug(dcTasmota) << "Publish received from Sonoff device:" << topic << payload; Device *dev = m_mqttChannels.key(channel); if (m_ipAddressParamTypeMap.contains(dev->deviceClassId())) { if (!topic.startsWith(channel->topicPrefix() + "/sonoff/POWER")) { return; } QString channelName = topic.split("/").last(); foreach (Device *child, myDevices()) { if (child->parentId() != dev->id()) { continue; } if (child->paramValue(m_channelParamTypeMap.value(child->deviceClassId())).toString() != channelName) { continue; } child->setStateValue(m_powerStateTypeMap.value(child->deviceClassId()), payload == "ON"); } } }