ESP8266 SHT21 sensor
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.

ESP-Weather.ino 8.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286
  1. #include <ESP8266WiFi.h>
  2. #include <WiFiClient.h>
  3. #include <ESP8266WebServer.h>
  4. #include <WiFiUdp.h>
  5. #include <SHT21.h>
  6. #include <vector>
  7. #include <WebSocketServer.h>
  8. #include <WiFiServer.h>
  9. #include <DNSServer.h>
  10. #include <WiFiManager.h>
  11. #include "storage.h"
  12. #define WEB_PORT 80
  13. #define BROADCAST_PORT 2390
  14. #define WEBSOCKET_PORT 2391
  15. #define NTP_PORT 2392
  16. #define DEFAULT_SSID "ESP-Weather"
  17. #define DEFAULT_PASS "testtest"
  18. #define MAX_BROADCAST_WAIT_TIME 550
  19. #define NTP_RETRY_TIMEOUT 2000
  20. // NTP-Client
  21. IPAddress timeServerIP;
  22. const char* ntpServerName = "time.nist.gov";
  23. const int NTP_PACKET_SIZE = 48;
  24. byte ntpPacketBuffer[NTP_PACKET_SIZE];
  25. WiFiUDP ntp;
  26. unsigned long timestamp = 0;
  27. unsigned long timeReceived = 0;
  28. unsigned long lastStorageTime = 0;
  29. byte storeAtBoot = 1;
  30. unsigned long lastNTP = 0;
  31. SHT21 SHT21;
  32. ESP8266WebServer server(WEB_PORT);
  33. WiFiServer serverSocket(WEBSOCKET_PORT);
  34. WebSocketServer webSocketServer;
  35. IPAddress broadcastIP;
  36. PersistentStorage storage;
  37. std::vector<IPAddress> vecClients;
  38. // UDP-Config
  39. #define UDP_PACKET_BUFFER_SIZE 255
  40. char packetBuffer[UDP_PACKET_BUFFER_SIZE];
  41. const char pingBuffer[] = "pingESP8266v0.1";
  42. const char echoBuffer[] = "echoESP8266v0.1";
  43. WiFiUDP Udp;
  44. unsigned long lastTime;
  45. bool waitingForReplies = false;
  46. const char* htmlBegin = "<html><head>\
  47. <title>Sysadmin</title>\
  48. <script src=\"http://hoegerle-home.de/sysAdmin/js/jquery-3.1.1.min.js\"></script>\
  49. <script src=\"http://hoegerle-home.de/sysAdmin/js/bootstrap.min.js\"></script>\
  50. <script src=\"http://hoegerle-home.de/sysAdmin/js/Chart.bundle.min.js\"></script>\
  51. <script src=\"http://hoegerle-home.de/sysAdmin/js/script.js\"></script>\
  52. <link rel=\"stylesheet\" href=\"http://hoegerle-home.de/sysAdmin/css/bootstrap.min.css\" />\
  53. </head><body>\
  54. <script type=\"text/javascript\">";
  55. const char* htmlEnd = "</script></body></html>";
  56. void handleRoot() {
  57. Serial.println("Sending UDP Broadcast...");
  58. // Send UDP broadcast to other modules
  59. Udp.beginPacket(broadcastIP, BROADCAST_PORT);
  60. Udp.write(pingBuffer);
  61. Udp.endPacket();
  62. // Start reply wait timer
  63. lastTime = millis();
  64. waitingForReplies = true;
  65. }
  66. void handleNotFound(){
  67. String message = "File Not Found\n\n";
  68. message += "URI: ";
  69. message += server.uri();
  70. message += "\nMethod: ";
  71. message += (server.method() == HTTP_GET)?"GET":"POST";
  72. message += "\nArguments: ";
  73. message += server.args();
  74. message += "\n";
  75. for (uint8_t i = 0; i < server.args(); i++){
  76. message += " " + server.argName(i) + ": " + server.arg(i) + "\n";
  77. }
  78. server.send(404, "text/plain", message);
  79. }
  80. void setup(void) {
  81. // Debugging
  82. Serial.begin(115200);
  83. Serial.println();
  84. Serial.println("ESP-Weather init...");
  85. //SHT21.begin();
  86. // The SHT library is simpy calling Wire.begin(), but the default
  87. // config does not match the pins i'm using (sda - 2; scl - 0)
  88. Wire.begin(2, 0);
  89. // Here you can override the WiFiManager configuration. Leave
  90. // the default autoConnect in use for the default behaviour.
  91. // To force the config portal, even though the module was connected before,
  92. // comment-out autoConnect and use startConfigPortal instead.
  93. WiFiManager wifiManager;
  94. wifiManager.autoConnect(DEFAULT_SSID, DEFAULT_PASS);
  95. //wifiManager.startConfigPortal(DEFAULT_SSID, DEFAULT_PASS);
  96. initMemory();
  97. storage = readMemory();
  98. // Wait for connection
  99. if (WiFi.status() != WL_CONNECTED) {
  100. while (WiFi.status() != WL_CONNECTED) {
  101. delay(500);
  102. Serial.print(".");
  103. }
  104. Serial.println("");
  105. }
  106. Serial.print("IP address: ");
  107. Serial.println(WiFi.localIP());
  108. broadcastIP = ~WiFi.subnetMask() | WiFi.gatewayIP();
  109. server.on("/", handleRoot);
  110. server.onNotFound(handleNotFound);
  111. server.begin();
  112. serverSocket.begin();
  113. // NTP-Client
  114. ntp.begin(NTP_PORT);
  115. WiFi.hostByName(ntpServerName, timeServerIP);
  116. lastNTP = millis();
  117. sendNTPpacket(timeServerIP);
  118. Udp.begin(BROADCAST_PORT);
  119. Serial.println("ESP-Weather ready!");
  120. }
  121. void loop(void){
  122. server.handleClient();
  123. // Websocket fuer Browser
  124. WiFiClient client = serverSocket.available();
  125. if (client.connected() && webSocketServer.handshake(client)) {
  126. Serial.println("Building WebSocket Response...");
  127. String json = "{\"H\":";
  128. json += String(SHT21.getHumidity());
  129. json += ",";
  130. json += "\"T\":";
  131. json += String(SHT21.getTemperature());
  132. json += ", \"EEPROM\" : [";
  133. for (int i = 0; i < storage.header.count; i++) {
  134. json += "{\"H\":";
  135. json += String(storage.data[i].humidity);
  136. json += ",";
  137. json += "\"T\":";
  138. json += String(storage.data[i].temperature);
  139. json += "}";
  140. if (i < storage.header.count - 1) {
  141. json += ",";
  142. }
  143. }
  144. json += "]}";
  145. Serial.println("WebSocket Response:");
  146. Serial.println(json);
  147. webSocketServer.sendData(json);
  148. client.flush();
  149. client.stop();
  150. }
  151. // NTP wiederholen falls keine Antwort
  152. if ((timestamp == 0) && ((millis() - lastNTP) > NTP_RETRY_TIMEOUT)) {
  153. Serial.println("NTP packet retry...");
  154. WiFi.hostByName(ntpServerName, timeServerIP);
  155. lastNTP = millis();
  156. sendNTPpacket(timeServerIP);
  157. }
  158. // NTP Paket vom Server erhalten
  159. if (ntp.parsePacket() >= NTP_PACKET_SIZE) {
  160. ntp.read(ntpPacketBuffer, NTP_PACKET_SIZE);
  161. unsigned long highWord = word(ntpPacketBuffer[40], ntpPacketBuffer[41]);
  162. unsigned long lowWord = word(ntpPacketBuffer[42], ntpPacketBuffer[43]);
  163. unsigned long secsSince1900 = highWord << 16 | lowWord;
  164. const unsigned long seventyYears = 2208988800UL;
  165. unsigned long epoch = secsSince1900 - seventyYears;
  166. timestamp = epoch;
  167. timeReceived = millis();
  168. Serial.print("Got NTP time: ");
  169. Serial.println(epoch);
  170. }
  171. // EEPROM-Schreiben jede Stunde
  172. if ((((((millis() - timeReceived) / 1000) + timestamp) % 3600) == 0)
  173. && (timestamp != 0) && (((millis() - lastStorageTime) > 100000) || storeAtBoot) ) {
  174. Serial.println("Storing new data packet...");
  175. lastStorageTime = millis();
  176. storeAtBoot = 0;
  177. if (storage.header.count < MAX_STORAGE) {
  178. storage.header.count++;
  179. } else {
  180. for(int i = 0; i < MAX_STORAGE - 1; i++) {
  181. storage.data[i] = storage.data[i+1];
  182. }
  183. }
  184. storage.data[storage.header.count - 1].temperature = SHT21.getTemperature();
  185. storage.data[storage.header.count - 1].humidity = SHT21.getHumidity();
  186. writeMemory(storage);
  187. }
  188. // UDP
  189. int packetSize = Udp.parsePacket();
  190. if (packetSize) {
  191. IPAddress remoteIp = Udp.remoteIP();
  192. // read the packet into packetBufffer
  193. int len = Udp.read(packetBuffer, UDP_PACKET_BUFFER_SIZE);
  194. if (len > 0) {
  195. packetBuffer[len] = 0;
  196. }
  197. Serial.print("Got UDP packet: ");
  198. Serial.println(packetBuffer);
  199. if (strcmp(packetBuffer, pingBuffer) == 0) {
  200. Serial.println("Broadcast");
  201. Udp.beginPacket(Udp.remoteIP(), Udp.remotePort());
  202. Udp.print(echoBuffer);
  203. Udp.endPacket();
  204. } else if((strcmp(packetBuffer, echoBuffer) == 0) && (waitingForReplies == true)) {
  205. vecClients.push_back(Udp.remoteIP());
  206. }
  207. }
  208. if (((millis() - lastTime) >= MAX_BROADCAST_WAIT_TIME) && (waitingForReplies == true)) {
  209. Serial.println("Timeout, sending response...");
  210. waitingForReplies = false;
  211. String message = htmlBegin;
  212. message += "var clients = Array(";
  213. message += "\"" + WiFi.localIP().toString() + "\"";
  214. if (vecClients.size() > 0) {
  215. message += ", ";
  216. }
  217. for (int i = 0; i < vecClients.size(); i++) {
  218. message += "\"" + vecClients[i].toString() + "\"";
  219. if (i < (vecClients.size() - 1)) {
  220. message += ", ";
  221. }
  222. }
  223. message += ");";
  224. message += htmlEnd;
  225. vecClients.clear();
  226. server.send(200, "text/html", message);
  227. }
  228. }
  229. void sendNTPpacket(IPAddress& address) {
  230. Serial.println("Sending NTP packet...");
  231. memset(ntpPacketBuffer, 0, NTP_PACKET_SIZE);
  232. ntpPacketBuffer[0] = 0b11100011; // LI, Version, Mode
  233. ntpPacketBuffer[1] = 0;
  234. ntpPacketBuffer[2] = 6;
  235. ntpPacketBuffer[3] = 0xEC;
  236. ntpPacketBuffer[12] = 49;
  237. ntpPacketBuffer[13] = 0x4E;
  238. ntpPacketBuffer[14] = 49;
  239. ntpPacketBuffer[15] = 52;
  240. ntp.beginPacket(address, 123);
  241. ntp.write(ntpPacketBuffer, NTP_PACKET_SIZE);
  242. ntp.endPacket();
  243. }