MqttClient.cpp 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. #include <stddef.h>
  2. #include <MqttClient.h>
  3. #include <TokenIterator.h>
  4. #include <UrlTokenBindings.h>
  5. #include <IntParsing.h>
  6. #include <ArduinoJson.h>
  7. #include <WiFiClient.h>
  8. #include <MiLightRadioConfig.h>
  9. MqttClient::MqttClient(Settings& settings, MiLightClient*& milightClient)
  10. : milightClient(milightClient),
  11. settings(settings),
  12. lastConnectAttempt(0)
  13. {
  14. String strDomain = settings.mqttServer();
  15. this->domain = new char[strDomain.length() + 1];
  16. strcpy(this->domain, strDomain.c_str());
  17. this->mqttClient = new PubSubClient(tcpClient);
  18. }
  19. MqttClient::~MqttClient() {
  20. mqttClient->disconnect();
  21. delete this->domain;
  22. }
  23. void MqttClient::begin() {
  24. #ifdef MQTT_DEBUG
  25. printf_P(
  26. PSTR("MqttClient - Connecting to: %s\nparsed:%s:%u\n"),
  27. settings._mqttServer.c_str(),
  28. settings.mqttServer().c_str(),
  29. settings.mqttPort()
  30. );
  31. #endif
  32. mqttClient->setServer(this->domain, settings.mqttPort());
  33. mqttClient->setCallback(
  34. [this](char* topic, byte* payload, int length) {
  35. this->publishCallback(topic, payload, length);
  36. }
  37. );
  38. reconnect();
  39. }
  40. bool MqttClient::connect() {
  41. char nameBuffer[30];
  42. sprintf_P(nameBuffer, PSTR("milight-hub-%u"), ESP.getChipId());
  43. #ifdef MQTT_DEBUG
  44. Serial.println(F("MqttClient - connecting"));
  45. #endif
  46. if (settings.mqttUsername.length() > 0) {
  47. return mqttClient->connect(
  48. nameBuffer,
  49. settings.mqttUsername.c_str(),
  50. settings.mqttPassword.c_str()
  51. );
  52. } else {
  53. return mqttClient->connect(nameBuffer);
  54. }
  55. }
  56. void MqttClient::reconnect() {
  57. if (lastConnectAttempt > 0 && (millis() - lastConnectAttempt) < MQTT_CONNECTION_ATTEMPT_FREQUENCY) {
  58. return;
  59. }
  60. if (! mqttClient->connected()) {
  61. if (connect()) {
  62. subscribe();
  63. #ifdef MQTT_DEBUG
  64. Serial.println(F("MqttClient - Successfully connected to MQTT server"));
  65. #endif
  66. } else {
  67. Serial.println(F("ERROR: Failed to connect to MQTT server"));
  68. }
  69. }
  70. lastConnectAttempt = millis();
  71. }
  72. void MqttClient::handleClient() {
  73. reconnect();
  74. mqttClient->loop();
  75. }
  76. void MqttClient::sendUpdate(const MiLightRemoteConfig& remoteConfig, uint16_t deviceId, uint16_t groupId, const char* update) {
  77. publish(settings.mqttUpdateTopicPattern, remoteConfig, deviceId, groupId, update);
  78. }
  79. void MqttClient::sendState(const MiLightRemoteConfig& remoteConfig, uint16_t deviceId, uint16_t groupId, const char* update) {
  80. publish(settings.mqttStateTopicPattern, remoteConfig, deviceId, groupId, update, true);
  81. }
  82. void MqttClient::subscribe() {
  83. String topic = settings.mqttTopicPattern;
  84. topic.replace(":device_id", "+");
  85. topic.replace(":group_id", "+");
  86. topic.replace(":device_type", "+");
  87. #ifdef MQTT_DEBUG
  88. printf_P(PSTR("MqttClient - subscribing to topic: %s\n"), topic.c_str());
  89. #endif
  90. mqttClient->subscribe(topic.c_str());
  91. }
  92. void MqttClient::publish(
  93. const String& _topic,
  94. const MiLightRemoteConfig &remoteConfig,
  95. uint16_t deviceId,
  96. uint16_t groupId,
  97. const char* message,
  98. const bool retain
  99. ) {
  100. if (_topic.length() == 0) {
  101. return;
  102. }
  103. String topic = _topic;
  104. MqttClient::bindTopicString(topic, remoteConfig, deviceId, groupId);
  105. #ifdef MQTT_DEBUG
  106. printf("MqttClient - publishing update to %s\n", topic.c_str());
  107. #endif
  108. mqttClient->publish(topic.c_str(), message, retain);
  109. }
  110. void MqttClient::publishCallback(char* topic, byte* payload, int length) {
  111. uint16_t deviceId = 0;
  112. uint8_t groupId = 0;
  113. const MiLightRemoteConfig* config = &FUT092Config;
  114. char cstrPayload[length + 1];
  115. cstrPayload[length] = 0;
  116. memcpy(cstrPayload, payload, sizeof(byte)*length);
  117. #ifdef MQTT_DEBUG
  118. printf("MqttClient - Got message on topic: %s\n%s\n", topic, cstrPayload);
  119. #endif
  120. char topicPattern[settings.mqttTopicPattern.length()];
  121. strcpy(topicPattern, settings.mqttTopicPattern.c_str());
  122. TokenIterator patternIterator(topicPattern, settings.mqttTopicPattern.length(), '/');
  123. TokenIterator topicIterator(topic, strlen(topic), '/');
  124. UrlTokenBindings tokenBindings(patternIterator, topicIterator);
  125. if (tokenBindings.hasBinding("device_id")) {
  126. deviceId = parseInt<uint16_t>(tokenBindings.get("device_id"));
  127. }
  128. if (tokenBindings.hasBinding("group_id")) {
  129. groupId = parseInt<uint16_t>(tokenBindings.get("group_id"));
  130. }
  131. if (tokenBindings.hasBinding("device_type")) {
  132. config = MiLightRemoteConfig::fromType(tokenBindings.get("device_type"));
  133. if (config == NULL) {
  134. Serial.println(F("MqttClient - ERROR: could not extract device_type from topic"));
  135. return;
  136. }
  137. } else {
  138. Serial.println(F("MqttClient - WARNING: could not find device_type token. Defaulting to FUT092.\n"));
  139. }
  140. StaticJsonBuffer<400> buffer;
  141. JsonObject& obj = buffer.parseObject(cstrPayload);
  142. #ifdef MQTT_DEBUG
  143. printf("MqttClient - device %04X, group %u\n", deviceId, groupId);
  144. #endif
  145. milightClient->prepare(config, deviceId, groupId);
  146. milightClient->update(obj);
  147. }
  148. inline void MqttClient::bindTopicString(
  149. String& topicPattern,
  150. const MiLightRemoteConfig& remoteConfig,
  151. const uint16_t deviceId,
  152. const uint16_t groupId
  153. ) {
  154. String deviceIdHex = String(deviceId, 16);
  155. deviceIdHex.toUpperCase();
  156. deviceIdHex = String("0x") + deviceIdHex;
  157. topicPattern.replace(":device_id", deviceIdHex);
  158. topicPattern.replace(":hex_device_id", deviceIdHex);
  159. topicPattern.replace(":dec_device_id", String(deviceId));
  160. topicPattern.replace(":group_id", String(groupId));
  161. topicPattern.replace(":device_type", remoteConfig.name);
  162. }