// SPDX-License-Identifier: GPL-2.0-or-later /* * Copyright (C) 2022 Thomas Basler and others */ #include "MqttPublishing.h" #include "MqttSettings.h" #include "NetworkSettings.h" #include MqttPublishingClass MqttPublishing; void MqttPublishingClass::init() { } void MqttPublishingClass::loop() { if (!MqttSettings.getConnected() || !Hoymiles.getRadio()->isIdle()) { return; } const CONFIG_T& config = Configuration.get(); if (millis() - _lastPublish > (config.Mqtt_PublishInterval * 1000)) { MqttSettings.publish("dtu/uptime", String(millis() / 1000)); MqttSettings.publish("dtu/ip", NetworkSettings.localIP().toString()); MqttSettings.publish("dtu/hostname", NetworkSettings.getHostname()); if (NetworkSettings.NetworkMode() == network_mode::WiFi) { MqttSettings.publish("dtu/rssi", String(WiFi.RSSI())); } // Loop all inverters for (uint8_t i = 0; i < Hoymiles.getNumInverters(); i++) { auto inv = Hoymiles.getInverterByPos(i); char buffer[sizeof(uint64_t) * 8 + 1]; snprintf(buffer, sizeof(buffer), "%0lx%08lx", ((uint32_t)((inv->serial() >> 32) & 0xFFFFFFFF)), ((uint32_t)(inv->serial() & 0xFFFFFFFF))); String subtopic = String(buffer); // Name MqttSettings.publish(subtopic + "/name", inv->name()); if (inv->DevInfo()->getLastUpdate() > 0) { // Bootloader Version MqttSettings.publish(subtopic + "/device/bootloaderversion", String(inv->DevInfo()->getFwBootloaderVersion())); // Firmware Version MqttSettings.publish(subtopic + "/device/fwbuildversion", String(inv->DevInfo()->getFwBuildVersion())); // Firmware Build DateTime char timebuffer[32]; const time_t t = inv->DevInfo()->getFwBuildDateTime(); std::strftime(timebuffer, sizeof(timebuffer), "%Y-%m-%d %H:%M:%S", gmtime(&t)); MqttSettings.publish(subtopic + "/device/fwbuilddatetime", String(timebuffer)); // Hardware part number MqttSettings.publish(subtopic + "/device/hwpartnumber", String(inv->DevInfo()->getHwPartNumber())); // Hardware version MqttSettings.publish(subtopic + "/device/hwversion", inv->DevInfo()->getHwVersion()); } if (inv->SystemConfigPara()->getLastUpdate() > 0) { // Limit MqttSettings.publish(subtopic + "/status/limit_relative", String(inv->SystemConfigPara()->getLimitPercent())); } MqttSettings.publish(subtopic + "/status/reachable", String(inv->isReachable())); MqttSettings.publish(subtopic + "/status/producing", String(inv->isProducing())); uint32_t lastUpdate = inv->Statistics()->getLastUpdate(); if (lastUpdate > 0 && lastUpdate != _lastPublishStats[i]) { _lastPublishStats[i] = lastUpdate; // Loop all channels for (uint8_t c = 0; c <= inv->Statistics()->getChannelCount(); c++) { for (uint8_t f = 0; f < sizeof(_publishFields); f++) { publishField(inv, c, _publishFields[f]); } } } yield(); } _lastPublish = millis(); } } void MqttPublishingClass::publishField(std::shared_ptr inv, uint8_t channel, uint8_t fieldId) { String topic = getTopic(inv, channel, fieldId); if (topic == "") { return; } MqttSettings.publish(topic, String(inv->Statistics()->getChannelFieldValue(channel, fieldId))); } String MqttPublishingClass::getTopic(std::shared_ptr inv, uint8_t channel, uint8_t fieldId) { if (!inv->Statistics()->hasChannelFieldValue(channel, fieldId)) { return String(""); } char buffer[sizeof(uint64_t) * 8 + 1]; snprintf(buffer, sizeof(buffer), "%0lx%08lx", ((uint32_t)((inv->serial() >> 32) & 0xFFFFFFFF)), ((uint32_t)(inv->serial() & 0xFFFFFFFF))); String invSerial = String(buffer); String chanName; if (channel == 0 && fieldId == FLD_PDC) { chanName = "powerdc"; } else { chanName = inv->Statistics()->getChannelFieldName(channel, fieldId); chanName.toLowerCase(); } return invSerial + "/" + String(channel) + "/" + chanName; }