forked from RoadkillUK/MAID2
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathMAID2.ino
460 lines (397 loc) · 29.5 KB
/
MAID2.ino
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
/**
* _____ ____ ____ _______ _
* |_ _||_ \ / _||_ __ \ / \
* | | | \/ | | |__) | / _ \
* _ | | | |\ /| | | __ / / ___ \
* | |__' | _| |_\/_| |_ _| | \ \_ _/ / \ \_
* `.____.' |_____||_____||____| |___||____| |____|
*
* ESP Energy Monitor v0.6.6 by Jorge Assunção
*
* Based on a project by timseebeck @ https://community.home-assistant.io/t/power-monitoring-with-an-xtm18s-and-mqtt/16316
* Remote Debug over Telnet by JoaoLopesF @ https://github.com/JoaoLopesF/ESP8266-RemoteDebug-Telnet
*
* See Github for instructions on how to use this code: https://github.com/jorgeassuncao/ESP8266-Energy-Monitor
*
* This program is free software; you can redistribute it and/or modify it under the terms of the
* GNU General Public License version 2 as published by the Free Software Foundation.
*
* You can change your personal user data (wifi access, MQTT server, etc) in the "config/userdata.h" file
*/
//************* INCLUDE LIBRARIES ************************************************************************
//********************************************************************************************************
#include <ESP8266WiFi.h>
#include <PubSubClient.h>
#include <WiFiClient.h>
#include <ESP8266WebServer.h>
#include <ESP8266mDNS.h>
#include "RemoteDebug.h"
#include "userdata.h"
#include <ArduinoOTA.h>
//************* PROJECT AND VERSION **********************************************************************
//********************************************************************************************************
const char* proj_ver = "MAID - Energy Monitor v0.6.6 (15/10/2017)"; // Project name and version
//************* CONFIG DEBUG *****************************************************************************
//********************************************************************************************************
RemoteDebug Debug; // Remote debug type
//************* CONFIG WEBSERVER *************************************************************************
//********************************************************************************************************
ESP8266WebServer server(80); // Webserver port
//************* CONFIG PINS *****************************************************************************
//********************************************************************************************************
#define DIGITAL_INPUT_SENSOR 12 // Digital input D6
#if TESTING
#else
int LED_pin = 2; // Internal LED in NodeMCU
#endif
#define BRIGHT 150 // Maximum LED intensity (1-500)
#define INHALE 1500 // Breathe-in time in milliseconds
#define PULSE INHALE*1000/BRIGHT // Pulse
#define REST 1000 // Pause between breathes
//************* CONFIG OTHER GLOBALS *********************************************************************
//********************************************************************************************************
volatile unsigned long pulseCount = 0; // Variable -
volatile unsigned long lastBlink = 0; // Variable -
double kwh; // Variable -
unsigned long lastSend; // Variable -
WiFiClient espClient; // Initiate wifi
PubSubClient client(espClient); // Initiate MQTT
long lastMsg = 0; // Variable -
char msg[50]; // Variable -
char wattString[7]; // Variable -
char kwhString[8]; // Variable -
double kwhaccum; // Variable -
char kwhaccumString[7]; // Variable -
float kwhReading; // Variable -
uint32_t mmLastTime = 0; // Variable -
uint32_t mmTimeSeconds = 0; // Variable -
int N = 1; // Variable - Number of times to loop
int runXTimes = 0; // Variable - Count times looped
byte mac[6]; // Variable - MAC address
char myBuffer[15]; // Variable - MAC string buffer
uint8_t MAC_array[6]; // Variable -
char MAC_char[18]; // Variable -
#if TESTING
float temptimer = 0;
#endif
//************* CONFIG WEBSERVER ROOT PAGE ***************************************************************
//********************************************************************************************************
String getPage(){ // Create webpage content
String page = "<!DOCTYPE html>";
page += "<html lang='en'>";
page += "<head>";
page += "<meta charset='utf-8'>";
page += "<meta http-equiv='X-UA-Compatible' content='IE=edge'>";
page += "<meta name='viewport' content='width=device-width, initial-scale=1'>";
page += "<meta http-equiv='refresh' content='15'/>";
page += "<link rel='shortcut icon' href=''>";
page += "<title>";
page += host_name;
page += "</title>";
page += "<link rel='stylesheet' href='//maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css' integrity='sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u' crossorigin='anonymous'>";
page += "<link rel='stylesheet' href='//maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap-theme.min.css' integrity='sha384-rHyoN1iRsVXV4nD0JutlnGaslCJuC7uwjduW9SVrLvRYooPp2bWYgmgJQIXwl/Sp' crossorigin='anonymous'>";
page += "<script src='//maxcdn.bootstrapcdn.com/bootstrap/3.3.7/js/bootstrap.min.js' integrity='sha384-Tc5IQib027qvyjSMfHjOMaLkfuWVxZxUPnCJA7l2mCWNIpG9mGCD8wGNIcPD7Txa' crossorigin='anonymous'></script>";
page += "<script src='https://ajax.googleapis.com/ajax/libs/jquery/3.2.0/jquery.min.js'></script>";
page += "</head>";
page += "<body>";
page += "<p></p>";
page += "<div class='container-fluid'>";
page += "<div class='row'>";
page += "<div class='col-md-12'>";
page += "<div class='jumbotron'>";
page += "<h2>";
page += proj_ver;
page += "</h2>";
page += "<p>This project uses a MQTT topic to store the accumulated kWh value, so that when there is a reboot, reset or power off the value won't be lost. The subscribed topic to get the accumulated kWh value is <small><em style='color: #ababab;'>";
page += mqtt_topic_sub_1;
page += "</em></small></p>";
page += "</div>";
page += "<div class='alert alert-dismissable alert-info'>";
page += "<button type='button' class='close' data-dismiss='alert' aria-hidden='true'>×</button>";
page += "<strong>Attention!</strong> This page auto-refreshes every 15 seconds.";
page += "</div>";
page += "</div>";
page += "</div>";
page += "<div class='row'>";
page += "<div class='col-md-4'>";
page += "<h3 class='text-primary'>Current Consumption</h3>";
page += "<p class='lead'>";
page += kwhString;
page += " kWh</p>";
page += "</div>";
page += "<div class='col-md-4'>";
page += "<h3 class='text-primary'>Current Power</h3>";
page += "<p class='lead'>";
page += wattString;
page += " Watt</p>";
page += "</div>";
page += "<div class='col-md-4'>";
page += "<h3 class='text-primary'>Accum Consumption</h3>";
page += "<p class='lead'>";
page += kwhaccumString;
page += " kWh (total)</p>";
page += "</div>";
page += "</div>";
page += "<div class='row'>";
page += "<div class='col-md-12'>";
page += "<div class='alert alert-dismissable alert-danger'>";
page += "<button type='button' class='close' data-dismiss='alert' aria-hidden='true'>×</button>";
page += "<h4>Warning!</h4>Sometimes the board does not restart correctly. If you can't get back to this page after one minute, you need to reset on the board directly on the fisical reset button. ";
page += "<button type='button' class='btn btn-warning btn-default'>RESET</button>";
page += "</div>";
page += "</div>";
page += "</div>";
page += "<div class='col-md-12'>";
page += "copyright <br/><br/>";
page += "</div>";
page += "</div>";
page += "</body>";
page += "</html>";
return page;
}
//************* SETUP WIFI SERVER ************************************************************************
//********************************************************************************************************
void handleRoot() { // Handle "root" page
server.send ( 200, "text/html", getPage() );
}
void handleNotFound(){ // Handle "not found" page
String message = "Page or File Not Found\n\n";
message += "URI: ";
message += server.uri();
message += "\nMethod: ";
message += (server.method() == HTTP_GET)?"GET":"POST";
message += "\nArguments: ";
message += server.args();
message += "\n";
for (uint8_t i=0; i<server.args(); i++){
message += " " + server.argName(i) + ": " + server.arg(i) + "\n";
}
server.send(404, "text/plain", message);
}
//************* SETUP WIFI *******************************************************************************
//********************************************************************************************************
void setup_wifi() {
delay(20);
/** TELNET **/
Debug.begin("ESP_Energy_Monitor_01_t"); // Initiaze the telnet server
Debug.setResetCmdEnabled(true); // Enable/disable (true/false) the reset command (true/false)
Debug.showTime(false); // Enable/disable (true/false) timestamps
Debug.showProfiler(false); // Enable/disable (true/false) Profiler - time between messages of Debug
Debug.showDebugLevel(false); // Enable/disable (true/false) debug levels
Debug.showColors(true); // Enable/disable (true/false) colors
Serial.println("- - - - - - - - - - - - - - - - - - - - - - - - - - -"); // Block separator to serial interface
Debug.println("- - - - - - - - - - - - - - - - - - - - - - - - - - -"); // Block separator to telnet debug interface
Serial.println(proj_ver); // Send project name and version to serial interface
Debug.println(proj_ver); // Send project name and version to telnet debug interface
Serial.println("- - - - - - - - - - - - - - - - - - - - - - - - - - -"); // Block separator to serial interface
Debug.println("- - - - - - - - - - - - - - - - - - - - - - - - - - -"); // Block separator to telnet debug interface
Serial.println(); // Send space to serial interface
Debug.println(); // Send space to telnet debug interface
Serial.println(); // Connecting to wifi network
Serial.print("Connecting to "); Serial.println(ssid); // Send network name to serial interface
Debug.printf("Connecting to "); Debug.println(ssid); // Send network name to telnet debug interface
WiFi.config(ip, dns, gateway, subnet); // Configure connection with IP, DNS, Gateway and subnet
WiFi.mode(WIFI_STA); // Switch to STA mode
WiFi.begin(ssid, password); // Start wifi connection with SSID and Password
WiFi.macAddress(mac); // Get MAC address of the node
while (WiFi.status() != WL_CONNECTED) { // Wait until connected to wifi
delay(500);
Serial.print(".");
}
Serial.println(); // Block space to serial interface
Debug.println(); // Block space to telnet debug interface
Serial.println("WiFi connected"); // Send successful connection to serial interface
Debug.println("WiFi connected"); // Send successful connection to telnet debug interface
Serial.print("IP address is "); Serial.println(WiFi.localIP()); // Send IP address to serial interface
Debug.printf("IP address is "); Debug.println(WiFi.localIP()); // Send IP address to telnet debug interface
sprintf(myBuffer,"%02X:%02X:%02X:%02X:%02X:%02X",mac[0],mac[1],mac[2],mac[3],mac[4],mac[5]); // Get MAC address
Serial.print("MAC address is "); Serial.println(myBuffer); // Send MAC address to serial interface
Debug.printf("MAC address is "); Debug.println(myBuffer); // Send MAC address to telnet debug interface
}
//************* READ MQTT TOPIC **************************************************************************
//********************************************************************************************************
void callback(char* topic, byte* payload, unsigned int length) {
if (runXTimes < N) { // Read the topic only N times
Serial.println(); // Block space to serial interface
Debug.println(); // Block space to telnet debug interface
Serial.print("Message arrived for topic ["); // Send text to serial interface
Debug.printf("Message arrived for topic ["); // Send text to telnet debug interface
Serial.print(topic); Serial.print("] "); // Send MQTT topic to serial interface
Debug.printf(topic); Debug.printf("] "); // Send MQTT topic to telnet debug interface
String value = ""; // Store MQTT topic inside string
for (int i=0;i<length;i++) {
value += (char)payload[i];
}
kwhReading = value.toFloat(); // Convert kwhReading to float
Serial.println(); // Block space to serial interface
Debug.println(); // Block space to telnet debug interface
Serial.print("Payload: "); Serial.println(kwhReading); // Send MQTT payload to serial interface
Debug.printf("Payload: "); Debug.println(kwhReading); // Send MQTT payload to telnet debug interface
Serial.println(); // Block space to serial interface
Debug.println(); // Block space to telnet debug interface
runXTimes++; // Increase loop count
}
}
//************* RECONNECT MQTT ***************************************************************************
//********************************************************************************************************
void reconnect() {
while (!client.connected()) { // Loop until reconnected
Serial.print("Attempting connection to MQTT server... "); // Send text to serial interface
Debug.printf("Attempting connection to MQTT server... "); // Send text to telnet debug interface
if (client.connect(host_name, mqtt_username, mqtt_password)) { // Connect to MQTT brocker
Serial.println(" connected!"); // Send text to serial interface
Debug.println(" connected!"); // Send text to telnet debug interface
client.subscribe(mqtt_topic_sub_1); // MQTT topic to subscribe
} else {
Serial.print("failed, rc="); // Send text to serial interface
Debug.printf("failed, rc="); // Send text to telnet debug interface
Serial.print(client.state()); // Send failure state to serial interface
//Debug.printf(client.state()); // Send failure state to telnet debug interface
Serial.println(" try again in 5 seconds"); // Send text to serial interface
Debug.println(" try again in 5 seconds!"); // Send text to telnet debug interface
delay(5000); // Wait 5 seconds before retrying
}
}
}
//************* ON PULSE *********************************************************************************
//********************************************************************************************************
void onPulse()
{
unsigned long newBlink = micros();
unsigned long interval = newBlink-lastBlink;
if (interval<debounce_interval) { // Sometimes we get interrupt on RISING
return;
}
kwh += 1.0 / (double)PULSE_FACTOR; // Every time there is a pulse, the energy consumption is 1 [pulse] / PULSE_FACTOR [pulses/kWh]
lastBlink = newBlink;
pulseCount++; // Accumulate the energy (it will be initialized again once MQTT message is sent)
Serial.print(interval); Serial.print(" | "); Serial.println(kwh);
Debug.print(interval); Debug.printf(" | "); Debug.println(kwh);
Serial.print("Pulse-"); Serial.println(pulseCount);
Debug.printf("Pulse-"); Debug.println(pulseCount);
}
//************* SETUP ************************************************************************************
//********************************************************************************************************
void setup()
{
#if TESTING
#else
pinMode(LED_pin, OUTPUT); // Configure internal LED pin as output.
#endif
WiFi.macAddress(MAC_array); // NEWNEWNEW
for (int iii = 0; iii < sizeof(MAC_array); ++iii){ // NEWNEWNEW
sprintf(MAC_char,"%s%02X:",MAC_char,MAC_array[iii]); // NEWNEWNEW
}
if (MDNS.begin("esp8266")) { //
Serial.println("MDNS responder started"); //
} //
server.on("/", handleRoot); // Serve root page
server.onNotFound(handleNotFound); // Serve page not found
server.begin(); // Start Webserver
Serial.println("HTTP server started"); // Send text to serial interface
Debug.println("HTTP server started"); // Send text to telnet debug interface
Serial.begin(115200); // Start serial interface
setup_wifi(); // Start wifi
client.setServer(mqtt_server, mqtt_port); //
client.setCallback(callback); //
// Use the internal pullup to be able to hook up this sketch directly to an energy meter with S0 output
// If no pullup is used, the reported usage will be too high because of the floating pin
if(LED_mode == 1) {
pinMode(DIGITAL_INPUT_SENSOR,INPUT);
} else {
pinMode(DIGITAL_INPUT_SENSOR,INPUT_PULLUP);
}
attachInterrupt(digitalPinToInterrupt(DIGITAL_INPUT_SENSOR), onPulse, RISING);
Debug.begin(host_name); // Start Telnet server
kwh = 0;
lastSend=millis();
//OTA SETUP
ArduinoOTA.setPort(OTAport);
ArduinoOTA.setHostname(host_name); // Hostname defaults to esp8266-[ChipID]
ArduinoOTA.setPassword((const char *)OTApassword); // No authentication by default
ArduinoOTA.onStart([]() {
Serial.println("Start");
});
ArduinoOTA.onEnd([]() {
Serial.println("\nEnd");
});
ArduinoOTA.onProgress([](unsigned int progress, unsigned int total) {
Serial.printf("Progress: %u%%\r", (progress / (total / 100)));
});
ArduinoOTA.onError([](ota_error_t error) {
Serial.printf("Error[%u]: ", error);
if (error == OTA_AUTH_ERROR) Serial.println("Auth Failed");
else if (error == OTA_BEGIN_ERROR) Serial.println("Begin Failed");
else if (error == OTA_CONNECT_ERROR) Serial.println("Connect Failed");
else if (error == OTA_RECEIVE_ERROR) Serial.println("Receive Failed");
else if (error == OTA_END_ERROR) Serial.println("End Failed");
});
ArduinoOTA.begin();
}
//************* LOOP *************************************************************************************
//********************************************************************************************************
void loop()
{
ArduinoOTA.handle(); // Check OTA Firmware Updates
server.handleClient(); // Handle http requests
#if TESTING
if(millis() - testpulse > temptimer) {
temptimer = millis();
onPulse();
}
#else
for (int ii=1;ii<BRIGHT;ii++){ // Breath in
digitalWrite(LED_pin, LOW); // LED on
delayMicroseconds(ii*10); // Wait
digitalWrite(LED_pin, HIGH); // LED off
delayMicroseconds(PULSE-ii*10); // Wait
delay(0); // Prevent watchdog firing
}
for (int ii=BRIGHT-1;ii>0;ii--){ // Breath out
digitalWrite(LED_pin, LOW); // LED on
delayMicroseconds(ii*10); // Wait
digitalWrite(LED_pin, HIGH); // LED off
delayMicroseconds(PULSE-ii*10); // Wait
ii--;
delay(0); // Prevent watchdog firing
}
delay(REST); // Pause before repeat
#endif
if (!client.connected()) { // If client disconnects...
Debug.println("Client Disconnected .... Reconnecting ....");
reconnect(); // ...connect again
}
client.loop();
unsigned long now = millis();
bool sendTime = now - lastSend > SEND_FREQUENCY; // Only send values at a maximum frequency
if (sendTime) {
Serial.println("- - - - - - - - - - - -"); // Block separator to serial interface
Debug.println("- - - - - - - - - - - -"); // Block separator to telnet debug interface
dtostrf(kwh, 2, 4, kwhString); // Convert Current kWh to string
client.publish(mqtt_topic_kwh,kwhString); // Publish Current kWh to MQTT topic
Serial.print("- Current kWh: "); Serial.println(kwhString); // Send Current kWh to serial interface
Debug.printf("- Current kWh: "); Debug.println(kwhString); // Send Current kWh to telnet debug interface
lastSend = now; // Update the send time after publishing
double watt = kwh * 1000.0 * 3600.0 / (double)SEND_FREQUENCY * 1000.0; // Calculate the power using the energy
dtostrf(watt, 5, 0, wattString); // Convert Current Watt to string
Serial.print("- Current Watt: "); Serial.println(wattString); // Send Current Watt to serial interface
Debug.printf("- Current Watt: "); Debug.println(wattString); // Send Current Watt to telnet debug interface
client.publish(mqtt_topic_watt,wattString); // Publish Current Watt to MQTT topic
kwhaccum = kwhReading + ((double)pulseCount/((double)PULSE_FACTOR)); // Calculate the accumulated energy since the begining
dtostrf(kwhaccum, 3, 3, kwhaccumString); // Convert Accum kWh (pulses) to string
client.publish(mqtt_topic_pulse,kwhaccumString, true); // Publish Accum kWh (pulses) to MQTT topic
Serial.print("- Accum kWh: "); Serial.println(kwhaccumString); // Send Accum kWh (pulses) to serial interface
Debug.printf("- Accum kWh: "); Debug.println(kwhaccumString); // Send Accum kWh (pulses) to telnet debug interface
kwh = 0; // Reset kWh count
String ipaddress = WiFi.localIP().toString(); // Create IP address string
char ipchar[ipaddress.length()+1];
ipaddress.toCharArray(ipchar,ipaddress.length()+1);
client.publish(mqtt_topic_ip,ipchar); // Publish IP address to MQTT topic
Serial.println(); // Block space to serial interface
Debug.println(); // Block space to telnet debug interface
}
Debug.handle(); // Remote debug over telnet
yield(); // Yielding
}
// END
//********************************************************************************************************