/* HOME ASSISTANT MODULE Copyright (C) 2017-2018 by Xose PĂ©rez */ #if HOMEASSISTANT_SUPPORT #include #include bool _haEnabled = false; bool _haSendFlag = false; // ----------------------------------------------------------------------------- // UTILS // ----------------------------------------------------------------------------- String _haFixName(String name) { for (unsigned char i=0; i 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 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(); output += "\n"; } output += " "; if (wrapJson) { output += "\"}"; } jsonBuffer.clear(); printer(output); } #if SENSOR_SUPPORT for (unsigned char i=0; i(); 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 _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