|
|
- /*
-
- HOME ASSISTANT MODULE
-
- Copyright (C) 2017-2018 by Xose Pérez <xose dot perez at gmail dot com>
-
- */
-
- #if HOMEASSISTANT_SUPPORT
-
- #include <ArduinoJson.h>
- #include <queue>
-
- bool _haEnabled = false;
- bool _haSendFlag = false;
-
- // -----------------------------------------------------------------------------
- // UTILS
- // -----------------------------------------------------------------------------
-
- String _haFixName(String name) {
- for (unsigned char i=0; i<name.length(); i++) {
- if (!isalnum(name.charAt(i))) name.setCharAt(i, '_');
- }
- return name;
- }
-
- // -----------------------------------------------------------------------------
- // SENSORS
- // -----------------------------------------------------------------------------
-
- #if SENSOR_SUPPORT
-
- void _haSendMagnitude(unsigned char i, const JsonObject& deviceConfig, JsonObject& config) {
-
- unsigned char type = magnitudeType(i);
- config["name"] = _haFixName(getSetting("hostname") + String(" ") + magnitudeTopic(type));
- config.set("platform", "mqtt");
- config["state_topic"] = mqttTopic(magnitudeTopicIndex(i).c_str(), false);
- config["unit_of_measurement"] = magnitudeUnits(type);
- config["uniq_id"] = getIdentifier() + "_" + type + "_" + String(i);
- config["device"] = deviceConfig;
-
- }
-
- void _haSendMagnitudes(const JsonObject& deviceConfig) {
-
- for (unsigned char i=0; i<magnitudeCount(); i++) {
-
- String topic = getSetting("haPrefix", HOMEASSISTANT_PREFIX) +
- "/sensor/" +
- getSetting("hostname") + "_" + String(i) +
- "/config";
-
- String output;
- if (_haEnabled) {
- DynamicJsonBuffer jsonBuffer;
- JsonObject& config = jsonBuffer.createObject();
- _haSendMagnitude(i, deviceConfig, config);
- config.printTo(output);
- jsonBuffer.clear();
- }
-
- mqttSendRaw(topic.c_str(), output.c_str());
- mqttSend(MQTT_TOPIC_STATUS, MQTT_STATUS_ONLINE, true);
-
- }
-
- }
-
- #endif // SENSOR_SUPPORT
-
- // -----------------------------------------------------------------------------
- // SWITCHES & LIGHTS
- // -----------------------------------------------------------------------------
-
- void _haSendSwitch(unsigned char i, const JsonObject& deviceConfig, JsonObject& config) {
-
- String name = getSetting("hostname");
- if (relayCount() > 1) {
- name += String("_") + String(i);
- }
-
- config.set("name", _haFixName(name));
- config.set("platform", "mqtt");
-
- config["uniq_id"] = getIdentifier() + "_switch_" + String(i);
- config["device"] = deviceConfig;
-
- if (relayCount()) {
- config["state_topic"] = mqttTopic(MQTT_TOPIC_RELAY, i, false);
- config["command_topic"] = mqttTopic(MQTT_TOPIC_RELAY, i, true);
- config["payload_on"] = String(HOMEASSISTANT_PAYLOAD_ON);
- config["payload_off"] = String(HOMEASSISTANT_PAYLOAD_OFF);
- config["availability_topic"] = mqttTopic(MQTT_TOPIC_STATUS, false);
- config["payload_available"] = String(HOMEASSISTANT_PAYLOAD_AVAILABLE);
- config["payload_not_available"] = String(HOMEASSISTANT_PAYLOAD_NOT_AVAILABLE);
- }
-
- #if LIGHT_PROVIDER != LIGHT_PROVIDER_NONE
-
- if (i == 0) {
-
- if (lightHasColor()) {
- config["brightness_state_topic"] = mqttTopic(MQTT_TOPIC_BRIGHTNESS, false);
- config["brightness_command_topic"] = mqttTopic(MQTT_TOPIC_BRIGHTNESS, true);
- config["rgb_state_topic"] = mqttTopic(MQTT_TOPIC_COLOR_RGB, false);
- config["rgb_command_topic"] = mqttTopic(MQTT_TOPIC_COLOR_RGB, true);
- config["color_temp_command_topic"] = mqttTopic(MQTT_TOPIC_MIRED, true);
- }
-
- if (lightChannels() > 3) {
- config["white_value_state_topic"] = mqttTopic(MQTT_TOPIC_CHANNEL, 3, false);
- config["white_value_command_topic"] = mqttTopic(MQTT_TOPIC_CHANNEL, 3, true);
- }
-
- }
-
- #endif // LIGHT_PROVIDER != LIGHT_PROVIDER_NONE
-
- }
-
- void _haSendSwitches(const JsonObject& deviceConfig) {
-
- #if (LIGHT_PROVIDER != LIGHT_PROVIDER_NONE) || (defined(ITEAD_SLAMPHER))
- String type = String("light");
- #else
- String type = String("switch");
- #endif
-
- for (unsigned char i=0; i<relayCount(); i++) {
-
- String topic = getSetting("haPrefix", HOMEASSISTANT_PREFIX) +
- "/" + type +
- "/" + getSetting("hostname") + "_" + String(i) +
- "/config";
-
- String output;
- if (_haEnabled) {
- DynamicJsonBuffer jsonBuffer;
- JsonObject& config = jsonBuffer.createObject();
- _haSendSwitch(i, deviceConfig, config);
- config.printTo(output);
- jsonBuffer.clear();
- }
-
- mqttSendRaw(topic.c_str(), output.c_str());
- mqttSend(MQTT_TOPIC_STATUS, MQTT_STATUS_ONLINE, true);
-
- }
-
- }
-
- // -----------------------------------------------------------------------------
-
- void _haDumpConfig(std::function<void(String&)> printer, bool wrapJson = false) {
-
- DynamicJsonBuffer deviceConfigJsonBuffer;
- JsonObject& deviceConfig = deviceConfigJsonBuffer.createObject();
- _haGetDeviceConfig(deviceConfig);
-
- #if (LIGHT_PROVIDER != LIGHT_PROVIDER_NONE) || (defined(ITEAD_SLAMPHER))
- String type = String("light");
- #else
- String type = String("switch");
- #endif
-
- for (unsigned char i=0; i<relayCount(); i++) {
-
- DynamicJsonBuffer jsonBuffer;
- JsonObject& config = jsonBuffer.createObject();
- _haSendSwitch(i, deviceConfig, config);
-
- String output;
- output.reserve(config.measureLength() + 32);
-
- if (wrapJson) {
- output += "{\"haConfig\": \"";
- }
-
- output += "\n\n" + type + ":\n";
- bool first = true;
-
- for (auto kv : config) {
- if (first) {
- output += " - ";
- first = false;
- } else {
- output += " ";
- }
- output += kv.key;
- output += ": ";
- output += kv.value.as<String>();
- output += "\n";
- }
- output += " ";
-
- if (wrapJson) {
- output += "\"}";
- }
-
- jsonBuffer.clear();
-
- printer(output);
-
- }
-
- #if SENSOR_SUPPORT
-
- for (unsigned char i=0; i<magnitudeCount(); i++) {
-
- DynamicJsonBuffer jsonBuffer;
- JsonObject& config = jsonBuffer.createObject();
- _haSendMagnitude(i, deviceConfig, config);
-
- String output;
- output.reserve(config.measureLength() + 32);
-
- if (wrapJson) {
- output += "{\"haConfig\": \"";
- }
-
- output += "\n\nsensor:\n";
- bool first = true;
-
- for (auto kv : config) {
- if (first) {
- output += " - ";
- first = false;
- } else {
- output += " ";
- }
- String value = kv.value.as<String>();
- value.replace("%", "'%'");
- output += kv.key;
- output += ": ";
- output += value;
- output += "\n";
- }
- output += " ";
-
- if (wrapJson) {
- output += "\"}";
- }
-
- jsonBuffer.clear();
-
- printer(output);
-
- }
-
- #endif
-
- deviceConfigJsonBuffer.clear();
- }
-
- void _haGetDeviceConfig(JsonObject& config) {
- String identifier = getIdentifier();
-
- config.createNestedArray("identifiers").add(identifier);
- config["name"] = _haFixName(getSetting("hostname"));
- config["manufacturer"] = String("Espurna");
- config["model"] = getBoardName();
- config["sw_version"] = String(APP_VERSION) + " (" + getCoreVersion() + ")";
- }
-
- void _haSend() {
-
- // Pending message to send?
- if (!_haSendFlag) return;
-
- // Are we connected?
- if (!mqttConnected()) return;
-
- DEBUG_MSG_P(PSTR("[HA] Sending autodiscovery MQTT message\n"));
-
- // Get common device config
- DynamicJsonBuffer jsonBuffer;
- JsonObject& deviceConfig = jsonBuffer.createObject();
- _haGetDeviceConfig(deviceConfig);
-
- // Send messages
- _haSendSwitches(deviceConfig);
- #if SENSOR_SUPPORT
- _haSendMagnitudes(deviceConfig);
- #endif
-
- jsonBuffer.clear();
- _haSendFlag = false;
-
- }
-
- void _haConfigure() {
- bool enabled = getSetting("haEnabled", HOMEASSISTANT_ENABLED).toInt() == 1;
- _haSendFlag = (enabled != _haEnabled);
- _haEnabled = enabled;
- _haSend();
- }
-
- #if WEB_SUPPORT
-
- std::queue<uint32_t> _ha_send_config;
-
- bool _haWebSocketOnReceive(const char * key, JsonVariant& value) {
- return (strncmp(key, "ha", 2) == 0);
- }
-
- void _haWebSocketOnSend(JsonObject& root) {
- root["haVisible"] = 1;
- root["haPrefix"] = getSetting("haPrefix", HOMEASSISTANT_PREFIX);
- root["haEnabled"] = getSetting("haEnabled", HOMEASSISTANT_ENABLED).toInt() == 1;
- }
-
- void _haWebSocketOnAction(uint32_t client_id, const char * action, JsonObject& data) {
- if (strcmp(action, "haconfig") == 0) {
- _ha_send_config.push(client_id);
- }
- }
-
- #endif
-
- #if TERMINAL_SUPPORT
-
- void _haInitCommands() {
-
- terminalRegisterCommand(F("HA.CONFIG"), [](Embedis* e) {
- _haDumpConfig([](String& data) {
- DEBUG_MSG(data.c_str());
- });
- DEBUG_MSG("\n");
- terminalOK();
- });
-
- terminalRegisterCommand(F("HA.SEND"), [](Embedis* e) {
- setSetting("haEnabled", "1");
- _haConfigure();
- #if WEB_SUPPORT
- wsSend(_haWebSocketOnSend);
- #endif
- terminalOK();
- });
-
- terminalRegisterCommand(F("HA.CLEAR"), [](Embedis* e) {
- setSetting("haEnabled", "0");
- _haConfigure();
- #if WEB_SUPPORT
- wsSend(_haWebSocketOnSend);
- #endif
- terminalOK();
- });
-
- }
-
- #endif
-
- // -----------------------------------------------------------------------------
-
- #if WEB_SUPPORT
- void _haLoop() {
- if (_ha_send_config.empty()) return;
-
- uint32_t client_id = _ha_send_config.front();
- _ha_send_config.pop();
-
- if (!wsConnected(client_id)) return;
-
- // TODO check wsConnected after each "printer" call?
- _haDumpConfig([client_id](String& output) {
- wsSend(client_id, output.c_str());
- yield();
- }, true);
- }
- #endif
-
- void haSetup() {
-
- _haConfigure();
-
- #if WEB_SUPPORT
- wsOnSendRegister(_haWebSocketOnSend);
- wsOnActionRegister(_haWebSocketOnAction);
- wsOnReceiveRegister(_haWebSocketOnReceive);
- espurnaRegisterLoop(_haLoop);
- #endif
-
- #if TERMINAL_SUPPORT
- _haInitCommands();
- #endif
-
- // On MQTT connect check if we have something to send
- mqttRegister([](unsigned int type, const char * topic, const char * payload) {
- if (type == MQTT_CONNECT_EVENT) _haSend();
- });
-
- // Main callbacks
- espurnaRegisterReload(_haConfigure);
-
- }
-
- #endif // HOMEASSISTANT_SUPPORT
|