ESP32 / ESP8266 & BME280 / SHT2x sensor with InfluxDB support
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.

html.cpp 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446
  1. /*
  2. * html.cpp
  3. *
  4. * ESP8266 / ESP32 Environmental Sensor
  5. *
  6. * ----------------------------------------------------------------------------
  7. * "THE BEER-WARE LICENSE" (Revision 42):
  8. * <xythobuz@xythobuz.de> wrote this file. As long as you retain this notice
  9. * you can do whatever you want with this stuff. If we meet some day, and you
  10. * think this stuff is worth it, you can buy me a beer in return. Thomas Buck
  11. * ----------------------------------------------------------------------------
  12. */
  13. #include <Arduino.h>
  14. #if defined(ARDUINO_ARCH_ESP8266)
  15. #include <ESP8266WiFi.h>
  16. #include <ESP8266WebServer.h>
  17. #include <ESP8266mDNS.h>
  18. #elif defined(ARDUINO_ARCH_ESP32)
  19. #include <WiFi.h>
  20. #include <WebServer.h>
  21. #include <ESPmDNS.h>
  22. #elif defined(ARDUINO_ARCH_AVR)
  23. #include <UnoWiFiDevEdSerial1.h>
  24. #include <WiFiLink.h>
  25. #endif
  26. #include "config.h"
  27. #include "DebugLog.h"
  28. #include "sensors.h"
  29. #include "servers.h"
  30. #include "memory.h"
  31. #include "relais.h"
  32. #include "moisture.h"
  33. #include "html.h"
  34. #if defined(ARDUINO_ARCH_AVR)
  35. #define ARDUINO_SEND_PARTIAL_PAGE() do { \
  36. size_t len = message.length(), off = 0; \
  37. while (off < len) { \
  38. if ((len - off) >= 50) { \
  39. client.write(message.c_str() + off, 50); \
  40. off += 50; \
  41. } else { \
  42. client.write(message.c_str() + off, len - off); \
  43. off = len; \
  44. } \
  45. } \
  46. message = ""; \
  47. } while (false);
  48. #else
  49. #define ARDUINO_SEND_PARTIAL_PAGE() while (false) { }
  50. #endif
  51. #if defined(ARDUINO_ARCH_ESP8266) || defined(ARDUINO_ARCH_ESP32)
  52. void handlePage(int mode, int id) {
  53. #else
  54. void handlePage(WiFiClient &client, int mode, int id) {
  55. #endif
  56. String message;
  57. message += F("<!DOCTYPE html>");
  58. message += F("<html><head>");
  59. message += F("<meta charset='utf-8'/>");
  60. message += F("<meta name='viewport' content='width=device-width, initial-scale=1'/>");
  61. message += F("<title>" ESP_PLATFORM_NAME " " NAME_OF_FEATURE "</title>");
  62. message += F("<style>");
  63. message += F(".log {\n");
  64. message += F( "max-height: 300px;\n");
  65. message += F( "padding: 0 1.0em;\n");
  66. message += F( "max-width: 1200px;\n");
  67. message += F( "margin: auto;\n");
  68. message += F( "margin-top: 1.5em;\n");
  69. message += F( "border: 1px dashed black;\n");
  70. message += F( "font-family: monospace;\n");
  71. message += F( "overflow-y: scroll;\n");
  72. message += F( "word-break: break-all;\n");
  73. message += F("}\n");
  74. message += F("#logbuf {\n");
  75. message += F( "white-space: break-spaces;\n");
  76. message += F("}\n");
  77. message += F("@media (prefers-color-scheme: dark) {");
  78. message += F( "body {");
  79. message += F( "background-color: black;");
  80. message += F( "color: white;");
  81. message += F( "}");
  82. message += F( "a:link { color: yellow; }");
  83. message += F( "a:visited { color: orange; }");
  84. message += F( ".log {\n");
  85. message += F( "border-color: white;");
  86. message += F( "}");
  87. message += F("}");
  88. message += F("</style>");
  89. message += F("</head><body>");
  90. message += F("<h1>" ESP_PLATFORM_NAME " " NAME_OF_FEATURE "</h1>");
  91. message += F("\n<p>\n");
  92. message += F("Version: ");
  93. message += ESP_ENV_VERSION;
  94. message += F("\n<br>\n");
  95. message += F("Build Date: ");
  96. message += __DATE__;
  97. message += F("\n<br>\n");
  98. message += F("Build Time: ");
  99. message += __TIME__;
  100. message += F("\n<br>\n");
  101. message += F("Location: ");
  102. message += SENSOR_LOCATION;
  103. message += F("\n<br>\n");
  104. message += F("ID: ");
  105. message += SENSOR_ID;
  106. #if defined(ARDUINO_ARCH_ESP8266) || defined(ARDUINO_ARCH_ESP32)
  107. message += F("\n<br>\n");
  108. message += F("MAC: ");
  109. message += WiFi.macAddress();
  110. #endif
  111. message += F("\n</p>\n");
  112. ARDUINO_SEND_PARTIAL_PAGE();
  113. #if defined(ARDUINO_ARCH_ESP8266)
  114. message += F("<p>");
  115. message += F("Reset reason: ");
  116. message += ESP.getResetReason();
  117. message += F("<br>");
  118. message += F("Free heap: ");
  119. message += String(ESP.getFreeHeap());
  120. message += F(" (");
  121. message += String(ESP.getHeapFragmentation());
  122. message += F("% fragmentation)");
  123. message += F("<br>");
  124. message += F("Free sketch space: ");
  125. message += String(ESP.getFreeSketchSpace());
  126. message += F("<br>");
  127. message += F("Flash chip real size: ");
  128. message += String(ESP.getFlashChipRealSize());
  129. if (ESP.getFlashChipSize() != ESP.getFlashChipRealSize()) {
  130. message += F("<br>");
  131. message += F("WARNING: sdk chip size (");
  132. message += (ESP.getFlashChipSize());
  133. message += F(") does not match!");
  134. }
  135. message += F("</p>");
  136. #elif defined(ARDUINO_ARCH_ESP32)
  137. message += F("<p>");
  138. message += F("Free heap: ");
  139. message += String(ESP.getFreeHeap() / 1024.0);
  140. message += F("k<br>");
  141. message += F("Free sketch space: ");
  142. message += String(ESP.getFreeSketchSpace() / 1024.0);
  143. message += F("k<br>");
  144. message += F("Flash chip size: ");
  145. message += String(ESP.getFlashChipSize() / 1024.0);
  146. message += F("k</p><hr>");
  147. #endif
  148. #ifdef ENABLE_BME280
  149. message += F("\n<p>\n");
  150. if (found_bme1) {
  151. message += F("BME280 Low:");
  152. message += F("\n<br>\n");
  153. message += F("Temperature: ");
  154. message += String(bme1_temp());
  155. message += F("\n<br>\n");
  156. message += F("Humidity: ");
  157. message += String(bme1_humid());
  158. message += F("\n<br>\n");
  159. message += F("Pressure: ");
  160. message += String(bme1_pressure());
  161. message += F("\n<br>\n");
  162. message += F("Offset: ");
  163. message += String(config.bme1_temp_off);
  164. message += F("\n<br>\n");
  165. message += F("<form method=\"GET\" action=\"/calibrate\">");
  166. message += F("<input type=\"text\" name=\"bme1\" placeholder=\"Real Temp.\">");
  167. message += F("<input type=\"submit\" value=\"Calibrate\">");
  168. message += F("</form>");
  169. } else {
  170. message += F("BME280 (low) not connected!");
  171. }
  172. message += F("\n</p><hr>\n");
  173. message += F("\n<p>\n");
  174. if (found_bme2) {
  175. message += F("BME280 High:");
  176. message += F("\n<br>\n");
  177. message += F("Temperature: ");
  178. message += String(bme2_temp());
  179. message += F("\n<br>\n");
  180. message += F("Humidity: ");
  181. message += String(bme2_humid());
  182. message += F("\n<br>\n");
  183. message += F("Pressure: ");
  184. message += String(bme2_pressure());
  185. message += F("\n<br>\n");
  186. message += F("Offset: ");
  187. message += String(config.bme2_temp_off);
  188. message += F("\n<br>\n");
  189. message += F("<form method=\"GET\" action=\"/calibrate\">");
  190. message += F("<input type=\"text\" name=\"bme2\" placeholder=\"Real Temp.\">");
  191. message += F("<input type=\"submit\" value=\"Calibrate\">");
  192. message += F("</form>");
  193. } else {
  194. message += F("BME280 (high) not connected!");
  195. }
  196. message += F("\n</p><hr>\n");
  197. #endif // ENABLE_BME280
  198. ARDUINO_SEND_PARTIAL_PAGE();
  199. message += F("\n<p>\n");
  200. if (found_sht) {
  201. message += F("SHT21:");
  202. message += F("\n<br>\n");
  203. message += F("Temperature: ");
  204. message += String(sht_temp());
  205. message += F("\n<br>\n");
  206. message += F("Humidity: ");
  207. message += String(sht_humid());
  208. message += F("\n<br>\n");
  209. message += F("Offset: ");
  210. message += String(config.sht_temp_off);
  211. message += F("\n<br>\n");
  212. message += F("<form method=\"GET\" action=\"/calibrate\">");
  213. message += F("<input type=\"text\" name=\"sht\" placeholder=\"Real Temp.\">");
  214. message += F("<input type=\"submit\" value=\"Calibrate\">");
  215. message += F("</form>");
  216. } else {
  217. message += F("SHT21 not connected!");
  218. }
  219. message += F("\n</p><hr>\n");
  220. #ifdef ENABLE_CCS811
  221. message += F("\n<p>\n");
  222. if (found_ccs1) {
  223. message += F("CCS811 Low:");
  224. message += F("\n<br>\n");
  225. message += F("eCO2: ");
  226. message += String(ccs1_eco2());
  227. message += F("ppm");
  228. message += F("\n<br>\n");
  229. message += F("TVOC: ");
  230. message += String(ccs1_tvoc());
  231. message += F("ppb");
  232. if (!ccs1_data_valid) {
  233. message += F("\n<br>\n");
  234. message += F("Data invalid (");
  235. message += String(ccs1_error_code);
  236. message += F(")!");
  237. }
  238. } else {
  239. message += F("CCS811 (Low) not connected!");
  240. }
  241. message += F("\n</p><hr>\n");
  242. message += F("\n<p>\n");
  243. if (found_ccs2) {
  244. message += F("CCS811 High:");
  245. message += F("\n<br>\n");
  246. message += F("eCO2: ");
  247. message += String(ccs2_eco2());
  248. message += F("ppm");
  249. message += F("\n<br>\n");
  250. message += F("TVOC: ");
  251. message += String(ccs2_tvoc());
  252. message += F("ppb");
  253. if (!ccs2_data_valid) {
  254. message += F("\n<br>\n");
  255. message += F("Data invalid (");
  256. message += String(ccs2_error_code);
  257. message += F(")!");
  258. }
  259. } else {
  260. message += F("CCS811 (High) not connected!");
  261. }
  262. message += F("\n</p><hr>\n");
  263. #endif // ENABLE_CCS811
  264. ARDUINO_SEND_PARTIAL_PAGE();
  265. #ifdef FEATURE_MOISTURE
  266. for (int i = 0; i < moisture_count(); i++) {
  267. int moisture = moisture_read(i);
  268. if (moisture < moisture_max()) {
  269. message += F("\n<p>\n");
  270. message += F("Sensor ");
  271. message += String(i + 1);
  272. message += F(":\n<br>\n");
  273. message += F("Moisture: ");
  274. message += String(moisture);
  275. message += F(" / ");
  276. message += String(moisture_max());
  277. message += F("\n</p>\n");
  278. }
  279. }
  280. if (moisture_count() <= 0) {
  281. message += F("\n<p>\n");
  282. message += F("No moisture sensors configured!");
  283. message += F("\n</p>\n");
  284. }
  285. ARDUINO_SEND_PARTIAL_PAGE();
  286. #endif // FEATURE_MOISTURE
  287. #ifdef FEATURE_RELAIS
  288. message += F("\n<p>\n");
  289. for (int i = 0; i < relais_count(); i++) {
  290. message += String(F("<a href=\"/on?id=")) + String(i) + String(F("\">Relais ")) + String(i) + String(F(" On (")) + relais_name(i) + String(F(")</a><br>\n"));
  291. message += String(F("<a href=\"/off?id=")) + String(i) + String(F("\">Relais ")) + String(i) + String(F(" Off (")) + relais_name(i) + String(F(")</a><br>\n"));
  292. }
  293. message += String(F("<a href=\"/on?id=")) + String(relais_count()) + String(F("\">All Relais On</a><br>\n"));
  294. message += String(F("<a href=\"/off?id=")) + String(relais_count()) + String(F("\">All Relais Off</a><br>\n"));
  295. message += F("</p>\n");
  296. if ((mode >= 0) && (mode <= 1)) {
  297. message += F("<p>");
  298. message += F("Turned Relais ");
  299. message += (id < relais_count()) ? String(id) : String(F("1-4"));
  300. message += (mode ? String(F(" On")) : String(F(" Off")));
  301. message += F("</p>\n");
  302. }
  303. message += F("\n<p>\n");
  304. for (int i = 0; i < relais_count(); i++) {
  305. message += String(F("Relais ")) + String(i) + String(F(" (")) + relais_name(i) + String(F(") = ")) + (relais_get(i) ? String(F("On")) : String(F("Off"))) + String(F("<br>\n"));
  306. }
  307. message += F("</p>\n");
  308. ARDUINO_SEND_PARTIAL_PAGE();
  309. #endif // FEATURE_RELAIS
  310. if (mode == 42) {
  311. message += F("<p>New calibration value saved!</p>\n");
  312. }
  313. #if ! defined(ARDUINO_ARCH_AVR)
  314. message += F("<p>");
  315. message += F("Try <a href=\"/update\">/update</a> for OTA firmware updates!");
  316. message += F("</p><p>");
  317. message += F("Try <a href=\"/reset\">/reset</a> to reset ESP!");
  318. message += F("</p>");
  319. #endif
  320. message += F("<p>");
  321. #ifdef ENABLE_INFLUXDB_LOGGING
  322. message += F("InfluxDB: ");
  323. message += INFLUXDB_DATABASE;
  324. message += F(" @ ");
  325. message += INFLUXDB_HOST;
  326. message += F(":");
  327. message += String(INFLUXDB_PORT);
  328. #else
  329. message += F("InfluxDB logging not enabled!");
  330. #endif
  331. message += F("</p>");
  332. message += F("<p>Uptime: ");
  333. message += String(millis() / 1000);
  334. message += F(" sec.</p>");
  335. #ifdef ENABLE_DEBUGLOG
  336. message += F("<hr><p>Debug Log:</p>");
  337. message += F("<div class='log'><pre id='logbuf'>");
  338. message += debug.getBuffer();
  339. message += F("</pre></div>");
  340. #endif // ENABLE_DEBUGLOG
  341. message += F("</body>");
  342. #ifdef ENABLE_WEBSOCKETS
  343. message += F("<script type='text/javascript'>");
  344. message += F("var socket = new WebSocket('ws://' + window.location.hostname + ':81');");
  345. message += F("socket.onmessage = function(e) {");
  346. message += F( "var log = document.getElementById('logbuf');");
  347. message += F( "var div = document.getElementsByClassName('log')[0];");
  348. message += F( "log.innerHTML += e.data.substring(4);");
  349. message += F( "if (log.innerHTML.length > (1024 * 1024)) {");
  350. message += F( "log.innerHTML = log.innerHTML.substring(1024 * 1024);");
  351. message += F( "}");
  352. message += F( "div.scrollTop = div.scrollHeight;");
  353. message += F("};");
  354. message += F("var hist = document.getElementsByClassName('log')[0];");
  355. message += F("hist.scrollTop = hist.scrollHeight;");
  356. message += F("</script>");
  357. #endif // ENABLE_WEBSOCKETS
  358. message += F("</html>");
  359. #if defined(ARDUINO_ARCH_ESP8266) || defined(ARDUINO_ARCH_ESP32)
  360. server.send(200, "text/html", message);
  361. #else
  362. ARDUINO_SEND_PARTIAL_PAGE();
  363. #endif
  364. }
  365. #if defined(ARDUINO_ARCH_ESP8266) || defined(ARDUINO_ARCH_ESP32)
  366. void handleReset() {
  367. String message;
  368. message += F("<!DOCTYPE html>");
  369. message += F("<html><head>");
  370. message += F("<meta charset='utf-8'/>");
  371. message += F("<meta name='viewport' content='width=device-width, initial-scale=1'/>");
  372. message += F("<meta http-equiv='refresh' content='10; URL=/'/>");
  373. message += F("<title>" ESP_PLATFORM_NAME " Environment Sensor</title>");
  374. message += F("<style>");
  375. message += F("@media (prefers-color-scheme: dark) {");
  376. message += F( "body {");
  377. message += F( "background-color: black;");
  378. message += F( "color: white;");
  379. message += F( "}");
  380. message += F( "a:link { color: yellow; }");
  381. message += F( "a:visited { color: orange; }");
  382. message += F("}");
  383. message += F("</style>");
  384. message += F("</head><body>");
  385. message += F("<p>Resetting in 2s...</p>");
  386. message += F("<p>Auto redirect in 10s. Please retry manually on error.</p>");
  387. message += F("<p>Go <a href=\"/\">back</a> to start.</p>");
  388. message += F("</body></html>");
  389. server.send(200, "text/html", message);
  390. delay(2000);
  391. ESP.restart();
  392. }
  393. #endif