/* DOMOTICZ MODULE Copyright (C) 2016-2019 by Xose PĂ©rez */ #include "domoticz.h" #if DOMOTICZ_SUPPORT #include "broker.h" #include "light.h" #include "mqtt.h" #include "relay.h" #include "rpc.h" #include "sensor.h" #include "ws.h" bool _dcz_enabled = false; std::bitset _dcz_relay_state; //------------------------------------------------------------------------------ // Private methods //------------------------------------------------------------------------------ unsigned int _domoticzIdx(unsigned char relayID, unsigned int defaultValue = 0) { return getSetting({"dczRelayIdx", relayID}, defaultValue); } int _domoticzRelay(unsigned int idx) { for (unsigned char relayID=0; relayID(), color["g"].as(), color["b"].as(), color["ww"].as(), color["cw"].as(), color["t"].as(), color["Level"].as(), idx ); // m field contains information about color mode (enum ColorMode from domoticz ColorSwitch.h): unsigned int cmode = color["m"]; if (cmode == 2) { // ColorModeWhite - WW,CW,temperature (t unused for now) if (lightChannels() < 2) return; lightChannel(0, color["ww"]); lightChannel(1, color["cw"]); } else if (cmode == 3 || cmode == 4) { // ColorModeRGB or ColorModeCustom if (lightChannels() < 3) return; lightChannel(0, color["r"]); lightChannel(1, color["g"]); lightChannel(2, color["b"]); // WARM WHITE (or MONOCHROME WHITE) and COLD WHITE are always sent. // Apply only when supported. if (lightChannels() > 3) { lightChannel(3, color["ww"]); } if (lightChannels() > 4) { lightChannel(4, color["cw"]); } } } // domoticz uses 100 as maximum value while we're using Light::BRIGHTNESS_MAX (unsigned char) lightBrightness((root["Level"].as() / 100.0) * Light::BRIGHTNESS_MAX); lightUpdate(true, mqttForward()); } #endif void _domoticzMqtt(unsigned int type, const char * topic, char * payload) { if (!_dcz_enabled) return; const String dczTopicOut = getSetting("dczTopicOut", DOMOTICZ_OUT_TOPIC); if (type == MQTT_CONNECT_EVENT) { // Subscribe to domoticz action topics mqttSubscribeRaw(dczTopicOut.c_str()); // Send relays state on connection #if RELAY_SUPPORT domoticzSendRelays(); #endif } if (type == MQTT_MESSAGE_EVENT) { // Check topic if (dczTopicOut.equals(topic)) { // Parse response DynamicJsonBuffer jsonBuffer(1024); JsonObject& root = jsonBuffer.parseObject(payload); if (!root.success()) { DEBUG_MSG_P(PSTR("[DOMOTICZ] Error parsing data\n")); return; } // IDX unsigned int idx = root["idx"]; #if LIGHT_PROVIDER != LIGHT_PROVIDER_NONE String stype = root["stype"]; String switchType = root["switchType"]; if ((_domoticzIdx(0) == idx) && (stype.startsWith("RGB") || (switchType.equals("Dimmer")))) { _domoticzLight(idx, root); } #endif int relayID = _domoticzRelay(idx); if (relayID >= 0) { unsigned char value = root["nvalue"]; DEBUG_MSG_P(PSTR("[DOMOTICZ] Received value %u for IDX %u\n"), value, idx); _domoticzStatus(relayID, value >= 1); } } } }; void _domoticzRelayConfigure(size_t size) { for (size_t n = 0; n < size; ++n) { _dcz_relay_state[n] = relayStatus(n); } } void _domoticzConfigure() { const bool enabled = getSetting("dczEnabled", 1 == DOMOTICZ_ENABLED); if (enabled != _dcz_enabled) _domoticzMqttSubscribe(enabled); #if RELAY_SUPPORT _domoticzRelayConfigure(relayCount()); #endif _dcz_enabled = enabled; } void _domoticzConfigCallback(const String& key, const String& value) { if (key.equals("relayDummy")) { _domoticzRelayConfigure(value.toInt()); return; } } void _domoticzBrokerCallback(const String& topic, unsigned char id, unsigned int value) { // Only process status messages for switches if (!topic.equals(MQTT_TOPIC_RELAY)) { return; } if (_domoticzStatus(id) == value) return; _dcz_relay_state[id] = value; domoticzSendRelay(id, value); } #if SENSOR_SUPPORT void domoticzSendMagnitude(unsigned char type, unsigned char index, double value, const char* buffer) { if (!_dcz_enabled) return; char key[15]; snprintf_P(key, sizeof(key), PSTR("dczMagnitude%d"), index); // Domoticz expects some additional data, dashboard might break otherwise. // https://www.domoticz.com/wiki/Domoticz_API/JSON_URL's#Barometer // TODO: Must send 'forecast' data. Default is last 3 hours: // https://github.com/domoticz/domoticz/blob/6027b1d9e3b6588a901de42d82f3a6baf1374cd1/hardware/I2C.cpp#L1092-L1193 // For now, just send invalid value. Consider simplifying sampling function and adding it here, with custom sampling time (3 hours, 6 hours, 12 hours etc.) if (MAGNITUDE_PRESSURE == type) { String svalue = buffer; svalue += ";-1"; domoticzSend(key, 0, svalue.c_str()); // Special case to allow us to use it with switches directly } else if (MAGNITUDE_DIGITAL == type) { int nvalue = (buffer[0] >= 48) ? (buffer[0] - 48) : 0; domoticzSend(key, nvalue, buffer); // https://www.domoticz.com/wiki/Domoticz_API/JSON_URL's#Humidity // nvalue contains HUM (relative humidity) // svalue contains HUM_STAT, one of consts below } else if (MAGNITUDE_HUMIDITY == type) { const char status = 48 + ( (value > 70) ? HUMIDITY_WET : (value > 45) ? HUMIDITY_COMFORTABLE : (value > 30) ? HUMIDITY_NORMAL : HUMIDITY_DRY ); char svalue[2] = {status, '\0'}; domoticzSend(key, static_cast(value), svalue); // Otherwise, send char string (nvalue is only for integers) } else { domoticzSend(key, 0, buffer); } } #endif // SENSOR_SUPPORT #if WEB_SUPPORT bool _domoticzWebSocketOnKeyCheck(const char * key, JsonVariant& value) { return (strncmp(key, "dcz", 3) == 0); } void _domoticzWebSocketOnVisible(JsonObject& root) { root["dczVisible"] = static_cast(haveRelaysOrSensors()); } void _domoticzWebSocketOnConnected(JsonObject& root) { root["dczEnabled"] = getSetting("dczEnabled", 1 == DOMOTICZ_ENABLED); root["dczTopicIn"] = getSetting("dczTopicIn", DOMOTICZ_IN_TOPIC); root["dczTopicOut"] = getSetting("dczTopicOut", DOMOTICZ_OUT_TOPIC); JsonArray& relays = root.createNestedArray("dczRelays"); for (unsigned char i=0; i void domoticzSend(const char * key, T nvalue, const char * svalue) { if (!_dcz_enabled) return; const auto idx = getSetting(key, 0); if (idx > 0) { char payload[128]; snprintf(payload, sizeof(payload), "{\"idx\": %d, \"nvalue\": %s, \"svalue\": \"%s\"}", idx, String(nvalue).c_str(), svalue); mqttSendRaw(getSetting("dczTopicIn", DOMOTICZ_IN_TOPIC).c_str(), payload); } } template void domoticzSend(const char * key, T nvalue) { domoticzSend(key, nvalue, ""); } void domoticzSendRelay(unsigned char relayID, bool status) { if (!_dcz_enabled) return; char buffer[15]; snprintf_P(buffer, sizeof(buffer), PSTR("dczRelayIdx%u"), relayID); domoticzSend(buffer, status ? "1" : "0"); } void domoticzSendRelays() { for (uint8_t relayID=0; relayID < relayCount(); relayID++) { domoticzSendRelay(relayID, relayStatus(relayID)); } } void domoticzSetup() { _domoticzConfigure(); #if WEB_SUPPORT wsRegister() .onVisible(_domoticzWebSocketOnVisible) .onConnected(_domoticzWebSocketOnConnected) .onKeyCheck(_domoticzWebSocketOnKeyCheck); #endif StatusBroker::Register(_domoticzBrokerCallback); ConfigBroker::Register(_domoticzConfigCallback); // Callbacks mqttRegister(_domoticzMqtt); espurnaRegisterReload(_domoticzConfigure); } bool domoticzEnabled() { return _dcz_enabled; } #endif