Settings.cpp 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. #include <Settings.h>
  2. #include <ArduinoJson.h>
  3. #include <FS.h>
  4. #include <IntParsing.h>
  5. #include <algorithm>
  6. #define PORT_POSITION(s) ( s.indexOf(':') )
  7. bool Settings::hasAuthSettings() {
  8. return adminUsername.length() > 0 && adminPassword.length() > 0;
  9. }
  10. bool Settings::isAutoRestartEnabled() {
  11. return _autoRestartPeriod > 0;
  12. }
  13. size_t Settings::getAutoRestartPeriod() {
  14. if (_autoRestartPeriod == 0) {
  15. return 0;
  16. }
  17. return std::max(_autoRestartPeriod, static_cast<size_t>(MINIMUM_RESTART_PERIOD));
  18. }
  19. void Settings::deserialize(Settings& settings, String json) {
  20. DynamicJsonBuffer jsonBuffer;
  21. JsonObject& parsedSettings = jsonBuffer.parseObject(json);
  22. settings.patch(parsedSettings);
  23. }
  24. void Settings::updateDeviceIds(JsonArray& arr) {
  25. if (arr.success()) {
  26. if (this->deviceIds) {
  27. delete this->deviceIds;
  28. }
  29. this->deviceIds = new uint16_t[arr.size()];
  30. this->numDeviceIds = arr.size();
  31. arr.copyTo(this->deviceIds, arr.size());
  32. }
  33. }
  34. void Settings::updateGatewayConfigs(JsonArray& arr) {
  35. if (arr.success()) {
  36. if (this->gatewayConfigs) {
  37. delete[] this->gatewayConfigs;
  38. }
  39. this->gatewayConfigs = new GatewayConfig*[arr.size()];
  40. this->numGatewayConfigs = arr.size();
  41. for (size_t i = 0; i < arr.size(); i++) {
  42. JsonArray& params = arr[i];
  43. if (params.success() && params.size() == 3) {
  44. this->gatewayConfigs[i] = new GatewayConfig(parseInt<uint16_t>(params[0]), params[1], params[2]);
  45. } else {
  46. Serial.print(F("Settings - skipped parsing gateway ports settings for element #"));
  47. Serial.println(i);
  48. }
  49. }
  50. }
  51. }
  52. void Settings::patch(JsonObject& parsedSettings) {
  53. if (parsedSettings.success()) {
  54. this->setIfPresent<String>(parsedSettings, "admin_username", adminUsername);
  55. this->setIfPresent(parsedSettings, "admin_password", adminPassword);
  56. this->setIfPresent(parsedSettings, "ce_pin", cePin);
  57. this->setIfPresent(parsedSettings, "csn_pin", csnPin);
  58. this->setIfPresent(parsedSettings, "reset_pin", resetPin);
  59. this->setIfPresent(parsedSettings, "packet_repeats", packetRepeats);
  60. this->setIfPresent(parsedSettings, "http_repeat_factor", httpRepeatFactor);
  61. this->setIfPresent(parsedSettings, "auto_restart_period", _autoRestartPeriod);
  62. this->setIfPresent(parsedSettings, "mqtt_server", _mqttServer);
  63. this->setIfPresent(parsedSettings, "mqtt_username", mqttUsername);
  64. this->setIfPresent(parsedSettings, "mqtt_password", mqttPassword);
  65. this->setIfPresent(parsedSettings, "mqtt_topic_pattern", mqttTopicPattern);
  66. this->setIfPresent(parsedSettings, "mqtt_update_topic_pattern", mqttUpdateTopicPattern);
  67. this->setIfPresent(parsedSettings, "mqtt_state_topic_pattern", mqttStateTopicPattern);
  68. this->setIfPresent(parsedSettings, "discovery_port", discoveryPort);
  69. this->setIfPresent(parsedSettings, "listen_repeats", listenRepeats);
  70. this->setIfPresent(parsedSettings, "state_flush_interval", stateFlushInterval);
  71. this->setIfPresent(parsedSettings, "mqtt_state_rate_limit", mqttStateRateLimit);
  72. if (parsedSettings.containsKey("radio_interface_type")) {
  73. this->radioInterfaceType = Settings::typeFromString(parsedSettings["radio_interface_type"]);
  74. }
  75. if (parsedSettings.containsKey("device_ids")) {
  76. JsonArray& arr = parsedSettings["device_ids"];
  77. updateDeviceIds(arr);
  78. }
  79. if (parsedSettings.containsKey("gateway_configs")) {
  80. JsonArray& arr = parsedSettings["gateway_configs"];
  81. updateGatewayConfigs(arr);
  82. }
  83. }
  84. }
  85. void Settings::load(Settings& settings) {
  86. if (SPIFFS.exists(SETTINGS_FILE)) {
  87. File f = SPIFFS.open(SETTINGS_FILE, "r");
  88. String settingsContents = f.readStringUntil(SETTINGS_TERMINATOR);
  89. f.close();
  90. deserialize(settings, settingsContents);
  91. } else {
  92. settings.save();
  93. }
  94. }
  95. String Settings::toJson(const bool prettyPrint) {
  96. String buffer = "";
  97. StringStream s(buffer);
  98. serialize(s, prettyPrint);
  99. return buffer;
  100. }
  101. void Settings::save() {
  102. File f = SPIFFS.open(SETTINGS_FILE, "w");
  103. if (!f) {
  104. Serial.println(F("Opening settings file failed"));
  105. } else {
  106. serialize(f);
  107. f.close();
  108. }
  109. }
  110. void Settings::serialize(Stream& stream, const bool prettyPrint) {
  111. DynamicJsonBuffer jsonBuffer;
  112. JsonObject& root = jsonBuffer.createObject();
  113. root["admin_username"] = this->adminUsername;
  114. root["admin_password"] = this->adminPassword;
  115. root["ce_pin"] = this->cePin;
  116. root["csn_pin"] = this->csnPin;
  117. root["reset_pin"] = this->resetPin;
  118. root["radio_interface_type"] = typeToString(this->radioInterfaceType);
  119. root["packet_repeats"] = this->packetRepeats;
  120. root["http_repeat_factor"] = this->httpRepeatFactor;
  121. root["auto_restart_period"] = this->_autoRestartPeriod;
  122. root["mqtt_server"] = this->_mqttServer;
  123. root["mqtt_username"] = this->mqttUsername;
  124. root["mqtt_password"] = this->mqttPassword;
  125. root["mqtt_topic_pattern"] = this->mqttTopicPattern;
  126. root["mqtt_update_topic_pattern"] = this->mqttUpdateTopicPattern;
  127. root["mqtt_state_topic_pattern"] = this->mqttStateTopicPattern;
  128. root["discovery_port"] = this->discoveryPort;
  129. root["listen_repeats"] = this->listenRepeats;
  130. root["state_flush_interval"] = this->stateFlushInterval;
  131. root["mqtt_state_rate_limit"] = this->mqttStateRateLimit;
  132. if (this->deviceIds) {
  133. JsonArray& arr = jsonBuffer.createArray();
  134. arr.copyFrom(this->deviceIds, this->numDeviceIds);
  135. root["device_ids"] = arr;
  136. }
  137. if (this->gatewayConfigs) {
  138. JsonArray& arr = jsonBuffer.createArray();
  139. for (size_t i = 0; i < this->numGatewayConfigs; i++) {
  140. JsonArray& elmt = jsonBuffer.createArray();
  141. elmt.add(this->gatewayConfigs[i]->deviceId);
  142. elmt.add(this->gatewayConfigs[i]->port);
  143. elmt.add(this->gatewayConfigs[i]->protocolVersion);
  144. arr.add(elmt);
  145. }
  146. root["gateway_configs"] = arr;
  147. }
  148. if (prettyPrint) {
  149. root.prettyPrintTo(stream);
  150. } else {
  151. root.printTo(stream);
  152. }
  153. }
  154. String Settings::mqttServer() {
  155. int pos = PORT_POSITION(_mqttServer);
  156. if (pos == -1) {
  157. return _mqttServer;
  158. } else {
  159. return _mqttServer.substring(0, pos);
  160. }
  161. }
  162. uint16_t Settings::mqttPort() {
  163. int pos = PORT_POSITION(_mqttServer);
  164. if (pos == -1) {
  165. return DEFAULT_MQTT_PORT;
  166. } else {
  167. return atoi(_mqttServer.c_str() + pos + 1);
  168. }
  169. }
  170. RadioInterfaceType Settings::typeFromString(const String& s) {
  171. if (s.equalsIgnoreCase("lt8900")) {
  172. return LT8900;
  173. } else {
  174. return nRF24;
  175. }
  176. }
  177. String Settings::typeToString(RadioInterfaceType type) {
  178. switch (type) {
  179. case LT8900:
  180. return "LT8900";
  181. case nRF24:
  182. default:
  183. return "nRF24";
  184. }
  185. }