Settings.cpp 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256
  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::updateGroupStateFields(JsonArray &arr) {
  53. if (arr.success()) {
  54. if (this->groupStateFields) {
  55. delete this->groupStateFields;
  56. }
  57. this->groupStateFields = new GroupStateField[arr.size()];
  58. this->numGroupStateFields = arr.size();
  59. for (size_t i = 0; i < arr.size(); i++) {
  60. String name = arr[i];
  61. name.toLowerCase();
  62. this->groupStateFields[i] = GroupStateFieldHelpers::getFieldByName(name.c_str());
  63. }
  64. }
  65. }
  66. void Settings::patch(JsonObject& parsedSettings) {
  67. if (parsedSettings.success()) {
  68. this->setIfPresent<String>(parsedSettings, "admin_username", adminUsername);
  69. this->setIfPresent(parsedSettings, "admin_password", adminPassword);
  70. this->setIfPresent(parsedSettings, "ce_pin", cePin);
  71. this->setIfPresent(parsedSettings, "csn_pin", csnPin);
  72. this->setIfPresent(parsedSettings, "reset_pin", resetPin);
  73. this->setIfPresent(parsedSettings, "packet_repeats", packetRepeats);
  74. this->setIfPresent(parsedSettings, "http_repeat_factor", httpRepeatFactor);
  75. this->setIfPresent(parsedSettings, "auto_restart_period", _autoRestartPeriod);
  76. this->setIfPresent(parsedSettings, "mqtt_server", _mqttServer);
  77. this->setIfPresent(parsedSettings, "mqtt_username", mqttUsername);
  78. this->setIfPresent(parsedSettings, "mqtt_password", mqttPassword);
  79. this->setIfPresent(parsedSettings, "mqtt_topic_pattern", mqttTopicPattern);
  80. this->setIfPresent(parsedSettings, "mqtt_update_topic_pattern", mqttUpdateTopicPattern);
  81. this->setIfPresent(parsedSettings, "mqtt_state_topic_pattern", mqttStateTopicPattern);
  82. this->setIfPresent(parsedSettings, "discovery_port", discoveryPort);
  83. this->setIfPresent(parsedSettings, "listen_repeats", listenRepeats);
  84. this->setIfPresent(parsedSettings, "state_flush_interval", stateFlushInterval);
  85. this->setIfPresent(parsedSettings, "mqtt_state_rate_limit", mqttStateRateLimit);
  86. this->setIfPresent(parsedSettings, "packet_repeat_throttle_threshold", packetRepeatThrottleThreshold);
  87. this->setIfPresent(parsedSettings, "packet_repeat_throttle_sensitivity", packetRepeatThrottleSensitivity);
  88. this->setIfPresent(parsedSettings, "packet_repeat_minimum", packetRepeatMinimum);
  89. if (parsedSettings.containsKey("radio_interface_type")) {
  90. this->radioInterfaceType = Settings::typeFromString(parsedSettings["radio_interface_type"]);
  91. }
  92. if (parsedSettings.containsKey("device_ids")) {
  93. JsonArray& arr = parsedSettings["device_ids"];
  94. updateDeviceIds(arr);
  95. }
  96. if (parsedSettings.containsKey("gateway_configs")) {
  97. JsonArray& arr = parsedSettings["gateway_configs"];
  98. updateGatewayConfigs(arr);
  99. }
  100. if (parsedSettings.containsKey("group_state_fields")) {
  101. JsonArray& arr = parsedSettings["group_state_fields"];
  102. updateGroupStateFields(arr);
  103. }
  104. }
  105. }
  106. void Settings::load(Settings& settings) {
  107. if (SPIFFS.exists(SETTINGS_FILE)) {
  108. File f = SPIFFS.open(SETTINGS_FILE, "r");
  109. String settingsContents = f.readStringUntil(SETTINGS_TERMINATOR);
  110. f.close();
  111. deserialize(settings, settingsContents);
  112. } else {
  113. settings.save();
  114. }
  115. }
  116. String Settings::toJson(const bool prettyPrint) {
  117. String buffer = "";
  118. StringStream s(buffer);
  119. serialize(s, prettyPrint);
  120. return buffer;
  121. }
  122. void Settings::save() {
  123. File f = SPIFFS.open(SETTINGS_FILE, "w");
  124. if (!f) {
  125. Serial.println(F("Opening settings file failed"));
  126. } else {
  127. serialize(f);
  128. f.close();
  129. }
  130. }
  131. void Settings::serialize(Stream& stream, const bool prettyPrint) {
  132. DynamicJsonBuffer jsonBuffer;
  133. JsonObject& root = jsonBuffer.createObject();
  134. root["admin_username"] = this->adminUsername;
  135. root["admin_password"] = this->adminPassword;
  136. root["ce_pin"] = this->cePin;
  137. root["csn_pin"] = this->csnPin;
  138. root["reset_pin"] = this->resetPin;
  139. root["radio_interface_type"] = typeToString(this->radioInterfaceType);
  140. root["packet_repeats"] = this->packetRepeats;
  141. root["http_repeat_factor"] = this->httpRepeatFactor;
  142. root["auto_restart_period"] = this->_autoRestartPeriod;
  143. root["mqtt_server"] = this->_mqttServer;
  144. root["mqtt_username"] = this->mqttUsername;
  145. root["mqtt_password"] = this->mqttPassword;
  146. root["mqtt_topic_pattern"] = this->mqttTopicPattern;
  147. root["mqtt_update_topic_pattern"] = this->mqttUpdateTopicPattern;
  148. root["mqtt_state_topic_pattern"] = this->mqttStateTopicPattern;
  149. root["discovery_port"] = this->discoveryPort;
  150. root["listen_repeats"] = this->listenRepeats;
  151. root["state_flush_interval"] = this->stateFlushInterval;
  152. root["mqtt_state_rate_limit"] = this->mqttStateRateLimit;
  153. root["packet_repeat_throttle_sensitivity"] = this->packetRepeatThrottleSensitivity;
  154. root["packet_repeat_throttle_threshold"] = this->packetRepeatThrottleThreshold;
  155. root["packet_repeat_minimum"] = this->packetRepeatMinimum;
  156. if (this->deviceIds) {
  157. JsonArray& arr = jsonBuffer.createArray();
  158. arr.copyFrom(this->deviceIds, this->numDeviceIds);
  159. root["device_ids"] = arr;
  160. }
  161. if (this->gatewayConfigs) {
  162. JsonArray& arr = jsonBuffer.createArray();
  163. for (size_t i = 0; i < this->numGatewayConfigs; i++) {
  164. JsonArray& elmt = jsonBuffer.createArray();
  165. elmt.add(this->gatewayConfigs[i]->deviceId);
  166. elmt.add(this->gatewayConfigs[i]->port);
  167. elmt.add(this->gatewayConfigs[i]->protocolVersion);
  168. arr.add(elmt);
  169. }
  170. root["gateway_configs"] = arr;
  171. }
  172. if (this->groupStateFields) {
  173. JsonArray& arr = jsonBuffer.createArray();
  174. for (size_t i = 0; i < this->numGroupStateFields; i++) {
  175. arr.add(GroupStateFieldHelpers::getFieldName(this->groupStateFields[i]));
  176. }
  177. root["group_state_fields"] = arr;
  178. }
  179. if (prettyPrint) {
  180. root.prettyPrintTo(stream);
  181. } else {
  182. root.printTo(stream);
  183. }
  184. }
  185. String Settings::mqttServer() {
  186. int pos = PORT_POSITION(_mqttServer);
  187. if (pos == -1) {
  188. return _mqttServer;
  189. } else {
  190. return _mqttServer.substring(0, pos);
  191. }
  192. }
  193. uint16_t Settings::mqttPort() {
  194. int pos = PORT_POSITION(_mqttServer);
  195. if (pos == -1) {
  196. return DEFAULT_MQTT_PORT;
  197. } else {
  198. return atoi(_mqttServer.c_str() + pos + 1);
  199. }
  200. }
  201. RadioInterfaceType Settings::typeFromString(const String& s) {
  202. if (s.equalsIgnoreCase("lt8900")) {
  203. return LT8900;
  204. } else {
  205. return nRF24;
  206. }
  207. }
  208. String Settings::typeToString(RadioInterfaceType type) {
  209. switch (type) {
  210. case LT8900:
  211. return "LT8900";
  212. case nRF24:
  213. default:
  214. return "nRF24";
  215. }
  216. }