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.

sysAdmin.ino 8.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322
  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 <EEPROM.h>
  10. #define NTP_PORT 2392
  11. #define MAX_WAIT_TIME 550
  12. #define EEPROM_SIZE 512
  13. // NTP-Client
  14. IPAddress timeServerIP;
  15. const char* ntpServerName = "time.nist.gov";
  16. const int NTP_PACKET_SIZE = 48;
  17. byte ntpPacketBuffer[NTP_PACKET_SIZE];
  18. WiFiUDP ntp;
  19. unsigned long timestamp = 0;
  20. unsigned long timeReceived = 0;
  21. unsigned long lastStorageTime = 0;
  22. byte storeAtBoot = 1;
  23. unsigned long lastNTP = 0;
  24. SHT21 SHT21;
  25. ESP8266WebServer server(80);
  26. WiFiServer serverSocket(2391);
  27. WebSocketServer webSocketServer;
  28. const char* ssid = "SysadminWLAN";
  29. const char* password = "sysadmin";
  30. IPAddress broadcastIP;
  31. struct __attribute__((__packed__)) Measurement {
  32. float temperature;
  33. float humidity;
  34. };
  35. struct __attribute__((__packed__)) Header {
  36. uint8_t count;
  37. uint16_t checksum;
  38. };
  39. #define MAX_STORAGE (EEPROM_SIZE - sizeof(Header)) / sizeof(Measurement)
  40. struct __attribute__((__packed__)) PersistentStorage {
  41. Measurement data[MAX_STORAGE];
  42. Header header;
  43. };
  44. PersistentStorage storage;
  45. void writeMemory(PersistentStorage &s) {
  46. Serial.println("write Memory");
  47. unsigned char* r = (unsigned char*) &s;
  48. uint16_t a = 0, b = 0;
  49. for(int i = 0; i < sizeof(PersistentStorage) - 2; i++) {
  50. a = (a + r[i]) % 255;
  51. b = (b + a) % 255;
  52. }
  53. s.header.checksum = (b << 8) | a;
  54. for(int i = 0; i < sizeof(PersistentStorage); i++) {
  55. EEPROM.write(i, r[i]);
  56. }
  57. EEPROM.commit();
  58. }
  59. PersistentStorage readMemory() {
  60. PersistentStorage s;
  61. unsigned char* r = (unsigned char*) &s;
  62. for(int i = 0; i < sizeof(PersistentStorage); i++) {
  63. r[i] = EEPROM.read(i);
  64. }
  65. uint16_t a = 0, b = 0;
  66. for(int i = 0; i < sizeof(PersistentStorage) - 2; i++) {
  67. a = (a + r[i]) % 255;
  68. b = (b + a) % 255;
  69. }
  70. if(s.header.checksum != ((b << 8) | a)) {
  71. Serial.print("Checksum error ");
  72. Serial.print(s.header.checksum);
  73. Serial.print(" ");
  74. Serial.println((b << 8) | a);
  75. s.header.count = 0;
  76. } else {
  77. Serial.println("Checksum ok");
  78. }
  79. return s;
  80. }
  81. std::vector<IPAddress> vecClients;
  82. // UDP-Config
  83. unsigned int localPort = 2390;
  84. char packetBuffer[255];
  85. char pingBuffer[] = "pingESP8266v0.1";
  86. char echoBuffer[] = "echoESP8266v0.1";
  87. WiFiUDP Udp;
  88. unsigned long lastTime;
  89. bool waitingForReplies = false;
  90. const char* htmlBegin = "<html>\
  91. <head>\
  92. <title>Sysadmin</title>\
  93. <script src=\"http://hoegerle-home.de/sysAdmin/js/jquery-3.1.1.min.js\"></script>\
  94. <script src=\"http://hoegerle-home.de/sysAdmin/js/bootstrap.min.js\"></script>\
  95. <script src=\"http://hoegerle-home.de/sysAdmin/js/Chart.bundle.min.js\"></script>\
  96. <script src=\"http://hoegerle-home.de/sysAdmin/js/script.js\"></script>\
  97. <link rel=\"stylesheet\" href=\"http://hoegerle-home.de/sysAdmin/css/bootstrap.min.css\" />\
  98. </head>\
  99. <body>\
  100. <script type=\"text/javascript\">";
  101. const char* htmlEnd = "</script>\
  102. </body>\
  103. </html>";
  104. // root-URL
  105. void handleRoot() {
  106. // send a reply, to the IP address and port that sent us the packet we received
  107. Udp.beginPacket(broadcastIP, 2390);
  108. Udp.write(pingBuffer);
  109. Udp.endPacket();
  110. // Timer starten
  111. lastTime = millis();
  112. waitingForReplies = true;
  113. Serial.println("broadcast send");
  114. }
  115. // URL nicht vorhanden
  116. void handleNotFound(){
  117. String message = "File Not Found\n\n";
  118. message += "URI: ";
  119. message += server.uri();
  120. message += "\nMethod: ";
  121. message += (server.method() == HTTP_GET)?"GET":"POST";
  122. message += "\nArguments: ";
  123. message += server.args();
  124. message += "\n";
  125. for (uint8_t i=0; i<server.args(); i++){
  126. message += " " + server.argName(i) + ": " + server.arg(i) + "\n";
  127. }
  128. server.send(404, "text/plain", message);
  129. }
  130. void setup(void) {
  131. SHT21.begin();
  132. EEPROM.begin(512);
  133. // Debugging
  134. Serial.begin(115200);
  135. WiFi.begin(ssid, password);
  136. Serial.println("");
  137. // Wait for connection
  138. while (WiFi.status() != WL_CONNECTED) {
  139. delay(500);
  140. Serial.print(".");
  141. }
  142. Serial.println("");
  143. Serial.print("Connected to ");
  144. Serial.println(ssid);
  145. Serial.print("IP address: ");
  146. Serial.println(WiFi.localIP());
  147. broadcastIP = ~WiFi.subnetMask() | WiFi.gatewayIP();
  148. storage = readMemory();
  149. server.on("/", handleRoot);
  150. server.onNotFound(handleNotFound);
  151. server.begin();
  152. serverSocket.begin();
  153. // NTP-Client
  154. ntp.begin(NTP_PORT);
  155. WiFi.hostByName(ntpServerName, timeServerIP);
  156. lastNTP = millis();
  157. sendNTPpacket(timeServerIP);
  158. Udp.begin(localPort);
  159. Serial.println("HTTP server and UDP started");
  160. }
  161. void loop(void){
  162. server.handleClient();
  163. // Websocket fuer Browser
  164. WiFiClient client = serverSocket.available();
  165. if(client.connected() && webSocketServer.handshake(client)) {
  166. String json = "{\"H\":";
  167. json += String(SHT21.getHumidity());
  168. json += ",";
  169. json += "\"T\":";
  170. json += String(SHT21.getTemperature());
  171. json += ", \"EEPROM\" : [";
  172. for(int i = 0; i < storage.header.count; i++) {
  173. json += "{\"H\":";
  174. json += String(storage.data[i].humidity);
  175. json += ",";
  176. json += "\"T\":";
  177. json += String(storage.data[i].temperature);
  178. json += "}";
  179. if(i < storage.header.count - 1) {
  180. json += ",";
  181. }
  182. }
  183. json += "]}";
  184. Serial.println(json);
  185. webSocketServer.sendData(json);
  186. client.flush();
  187. client.stop();
  188. }
  189. // NTP wiederholen falls keine Antwort
  190. if((timestamp == 0) && ((millis() - lastNTP) > 2000)) {
  191. WiFi.hostByName(ntpServerName, timeServerIP);
  192. lastNTP = millis();
  193. sendNTPpacket(timeServerIP);
  194. }
  195. // NTP Paket vom Server erhalten
  196. if(ntp.parsePacket() >= NTP_PACKET_SIZE) {
  197. ntp.read(ntpPacketBuffer, NTP_PACKET_SIZE);
  198. unsigned long highWord = word(ntpPacketBuffer[40], ntpPacketBuffer[41]);
  199. unsigned long lowWord = word(ntpPacketBuffer[42], ntpPacketBuffer[43]);
  200. unsigned long secsSince1900 = highWord << 16 | lowWord;
  201. const unsigned long seventyYears = 2208988800UL;
  202. unsigned long epoch = secsSince1900 - seventyYears;
  203. timestamp = epoch;
  204. timeReceived = millis();
  205. Serial.println(epoch);
  206. }
  207. // EEPROM-Schreiben jede Stunde
  208. if((((((millis() - timeReceived) / 1000) + timestamp) % 3600) == 0) && (timestamp != 0) && (((millis() - lastStorageTime) > 100000) || storeAtBoot) ) {
  209. lastStorageTime = millis();
  210. storeAtBoot = 0;
  211. if(storage.header.count < MAX_STORAGE) {
  212. storage.header.count++;
  213. } else {
  214. for(int i = 0; i < MAX_STORAGE - 1; i++) {
  215. storage.data[i] = storage.data[i+1];
  216. }
  217. }
  218. storage.data[storage.header.count-1].temperature = SHT21.getTemperature();
  219. storage.data[storage.header.count-1].humidity = SHT21.getHumidity();
  220. writeMemory(storage);
  221. }
  222. // UDP
  223. int packetSize = Udp.parsePacket();
  224. if(packetSize) {
  225. IPAddress remoteIp = Udp.remoteIP();
  226. // read the packet into packetBufffer
  227. int len = Udp.read(packetBuffer, 255);
  228. if (len > 0) {
  229. packetBuffer[len] = 0;
  230. }
  231. Serial.println(packetBuffer);
  232. if(strcmp(packetBuffer, pingBuffer) == 0) {
  233. Serial.println("Broadcast");
  234. Udp.beginPacket(Udp.remoteIP(), Udp.remotePort());
  235. Udp.print(echoBuffer);
  236. Udp.endPacket();
  237. } else if((strcmp(packetBuffer, echoBuffer) == 0) && (waitingForReplies == true)) {
  238. vecClients.push_back(Udp.remoteIP());
  239. }
  240. }
  241. if(((millis() - lastTime) >= MAX_WAIT_TIME) && (waitingForReplies == true)) {
  242. Serial.println("waiting done");
  243. waitingForReplies = false;
  244. String message = htmlBegin;
  245. message += "var clients = Array(";
  246. message += "\"" + WiFi.localIP().toString() + "\"";
  247. if(vecClients.size() > 0) {
  248. message += ", ";
  249. }
  250. for(int i = 0; i < vecClients.size(); i++) {
  251. message += "\"" + vecClients[i].toString() + "\"";
  252. if(i < (vecClients.size() - 1)) {
  253. message += ", ";
  254. }
  255. }
  256. message += ");";
  257. message += htmlEnd;
  258. vecClients.clear();
  259. server.send(200, "text/html", message);
  260. }
  261. }
  262. void sendNTPpacket(IPAddress& address) {
  263. Serial.println("sending NTP packet...");
  264. memset(ntpPacketBuffer, 0, NTP_PACKET_SIZE);
  265. ntpPacketBuffer[0] = 0b11100011; // LI, Version, Mode
  266. ntpPacketBuffer[1] = 0;
  267. ntpPacketBuffer[2] = 6;
  268. ntpPacketBuffer[3] = 0xEC;
  269. ntpPacketBuffer[12] = 49;
  270. ntpPacketBuffer[13] = 0x4E;
  271. ntpPacketBuffer[14] = 49;
  272. ntpPacketBuffer[15] = 52;
  273. ntp.beginPacket(address, 123);
  274. ntp.write(ntpPacketBuffer, NTP_PACKET_SIZE);
  275. ntp.endPacket();
  276. }