Fork of the espurna firmware for `mhsw` switches
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

131 lines
3.5 KiB

  1. /*
  2. DOMOTICZ MODULE
  3. Copyright (C) 2016-2017 by Xose Pérez <xose dot perez at gmail dot com>
  4. */
  5. #if DOMOTICZ_SUPPORT
  6. #include <ArduinoJson.h>
  7. bool _dcz_enabled = false;
  8. unsigned long _dcz_skip_time = 0;
  9. unsigned long _dcz_last_idx = 0;
  10. unsigned long _dcz_last_time = 0;
  11. //------------------------------------------------------------------------------
  12. // Private methods
  13. //------------------------------------------------------------------------------
  14. int _domoticzRelay(unsigned int idx) {
  15. for (int relayID=0; relayID<relayCount(); relayID++) {
  16. if (domoticzIdx(relayID) == idx) {
  17. return relayID;
  18. }
  19. }
  20. return -1;
  21. }
  22. bool _domoticzSkip(unsigned long idx) {
  23. if (idx == _dcz_last_idx && (millis() - _dcz_last_time < _dcz_skip_time)) return true;
  24. _dcz_last_idx = idx;
  25. _dcz_last_time = millis();
  26. return false;
  27. }
  28. void _domoticzMqtt(unsigned int type, const char * topic, const char * payload) {
  29. String dczTopicOut = getSetting("dczTopicOut", DOMOTICZ_OUT_TOPIC);
  30. if (type == MQTT_CONNECT_EVENT) {
  31. mqttSubscribeRaw(dczTopicOut.c_str());
  32. }
  33. if (type == MQTT_MESSAGE_EVENT) {
  34. if (!_dcz_enabled) return;
  35. // Check topic
  36. if (dczTopicOut.equals(topic)) {
  37. // Parse response
  38. DynamicJsonBuffer jsonBuffer;
  39. JsonObject& root = jsonBuffer.parseObject((char *) payload);
  40. if (!root.success()) {
  41. DEBUG_MSG_P(PSTR("[DOMOTICZ] Error parsing data\n"));
  42. return;
  43. }
  44. // IDX
  45. unsigned long idx = root["idx"];
  46. int relayID = _domoticzRelay(idx);
  47. if (relayID >= 0) {
  48. // Skip message if recursive
  49. if (_domoticzSkip(idx)) return;
  50. unsigned long value = root["nvalue"];
  51. DEBUG_MSG_P(PSTR("[DOMOTICZ] Received value %d for IDX %d\n"), value, idx);
  52. relayStatus(relayID, value == 1);
  53. }
  54. }
  55. }
  56. };
  57. //------------------------------------------------------------------------------
  58. // Public API
  59. //------------------------------------------------------------------------------
  60. template<typename T> void domoticzSend(const char * key, T nvalue, const char * svalue) {
  61. if (!_dcz_enabled) return;
  62. unsigned int idx = getSetting(key).toInt();
  63. if (idx > 0) {
  64. // Skip message if recursive
  65. if (_domoticzSkip(idx)) return;
  66. char payload[128];
  67. snprintf(payload, sizeof(payload), "{\"idx\": %d, \"nvalue\": %s, \"svalue\": \"%s\"}", idx, String(nvalue).c_str(), svalue);
  68. mqttSendRaw(getSetting("dczTopicIn", DOMOTICZ_IN_TOPIC).c_str(), payload);
  69. }
  70. }
  71. template<typename T> void domoticzSend(const char * key, T nvalue) {
  72. domoticzSend(key, nvalue, "");
  73. }
  74. void domoticzSendRelay(unsigned int relayID) {
  75. if (!_dcz_enabled) return;
  76. char buffer[15];
  77. snprintf_P(buffer, sizeof(buffer), PSTR("dczRelayIdx%d"), relayID);
  78. domoticzSend(buffer, relayStatus(relayID) ? "1" : "0");
  79. }
  80. int domoticzIdx(unsigned int relayID) {
  81. char buffer[15];
  82. snprintf_P(buffer, sizeof(buffer), PSTR("dczRelayIdx%d"), relayID);
  83. return getSetting(buffer).toInt();
  84. }
  85. void domoticzConfigure() {
  86. _dcz_enabled = getSetting("dczEnabled", DOMOTICZ_ENABLED).toInt() == 1;
  87. _dcz_skip_time = 1000 * getSetting("dczSkip", DOMOTICZ_SKIP_TIME).toInt();
  88. }
  89. void domoticzSetup() {
  90. domoticzConfigure();
  91. mqttRegister(_domoticzMqtt);
  92. }
  93. bool domoticzEnabled() {
  94. return _dcz_enabled;
  95. }
  96. #endif