-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy path0_arduino_code.ino
508 lines (442 loc) · 13 KB
/
0_arduino_code.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
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
#include <Arduino.h>
// here the parameters for my device are set: name, room, verbose mode
#include "device_setup.h"
#include "TM_Helper.h"
#include "TM_ESP32_Class.h"
auto my_esp32 = TM_ESP32_Class(myVerbose);
#ifdef TM_LOAD_DEVICE_INFLUXDB
#include "TM_InfluxDB_Class.h"
#include <InfluxDbClient.h>
auto my_influx = TM_Influx_Class(myVerbose);
Point influx_data_set("Raumklima"); // Table Name
uint32_t timestampLastTimeSync;
uint8_t hour;
uint8_t minute;
#endif
#ifdef TM_LOAD_DEVICE_BME280
#include "TM_BME280_Class.h"
auto my_sensor_bme280 = TM_BME280_Class(myVerbose);
float *data_bme280;
#endif
#ifdef TM_LOAD_DEVICE_BME680
// for BME680 in LP mode, the loop sleep time must be 3s!
#include "TM_BME680_Class.h"
auto my_sensor_bme680 = TM_BME680_Class(myVerbose);
float *data_bme680;
#endif
#ifdef TM_LOAD_DEVICE_BH1750
#include "TM_BH1750_Class.h"
auto my_sensor_illuminance = TM_BH1750_Class(myVerbose);
float data_lux;
#endif
#ifdef TM_LOAD_DEVICE_MHZ19
#include "TM_MH_Z19_Class.h"
// Attention from EPS32 point of view the RX and TX are swapped, RX(MH-Z19)=TX(ESP32) and vice versa
// new layout
// #define TM_MHZ19_PIN_RX 16
// #define TM_MHZ19_PIN_TX 17
// old KiTa layout
// #define TM_MHZ19_PIN_RX 14
// #define TM_MHZ19_PIN_TX 13
auto my_sensor_CO2 = TM_MH_Z19_Class(TM_MHZ19_PIN_RX, TM_MHZ19_PIN_TX, myVerbose);
uint8_t count_same_CO2_values = 0; // MH-Z19 and WiFi/InfluxDB have a problem, that when both are enabled, the MH-Z19 from time to time is stuck and returns always a value of 380
int data_CO2 = 0;
#endif
#ifdef TM_LOAD_DEVICE_4_DIGIT
#include "TM_4DigitDisplay_Class.h"
auto my_display_4digit = TM_4DigitDisplay_Class(32, 33, myVerbose);
int my_display_4digit_loop_delay = 0; // ms
#endif
#ifdef TM_LOAD_DEVICE_OLED_128X32
#include "TM_OLED_Class.h"
auto my_display_oled = TM_OLED_128x32_Class(myVerbose);
#endif
#ifdef TM_LOAD_DEVICE_OLED_128X64
#include "TM_OLED_Class.h"
auto my_display_oled = TM_OLED_128x64_Class(myVerbose);
#endif
#ifdef TM_LOAD_DEVICE_LED_RING
#include "TM_LED_Ring_Class.h"
auto my_display_led_rbg_ring = TM_LED_Ring_Class(15, 8, myVerbose);
#endif
#ifdef TM_LOAD_DEVICE_LED_KY_016
#include "TM_LED_KY_016_Class.h"
// #include <analogWrite.h>
// #include "RGBLed.h"
auto my_display_led_rbg_single = TM_LED_KY_016_Class(5, 18, 19, myVerbose);
#endif
// variables
uint8_t loopNum = 0;
uint32_t timeStart;
float data_to_display = 0;
const float value_min_CO2 = 400;
const float value_max_CO2 = 1150;
// new: 400-1150: 3 colors for 750 values -> blue for 650-900
// old: 400-1000: 3 colors for 600 values -> blue for 600-800
bool display_shall_sleep = false;
//
//
// II. Setup
//
//
void setup()
{
Serial.begin(115200);
// Fuck: This sometimes results in an endless wait if not connected to PC
// while (!Serial)
// ; // time to get serial running
// Serial.flush();
#if !defined(TM_LOAD_DEVICE_LED_RING)
my_esp32.underclocking(); // underclocking breaks Adafruit_NeoPixel !!!
#endif
my_esp32.init();
#ifdef TM_LOAD_DEVICE_INFLUXDB
my_influx.connect_wifi(my_device_name);
#if defined(TM_DISPLAY_TIME) || defined(TM_HOUR_SLEEP) && defined(TM_HOUR_WAKE)
my_influx.sync_time();
timestampLastTimeSync = getTimestamp();
#endif
my_influx.connect_influxdb();
influx_data_set.addTag("device", my_device_name);
influx_data_set.addTag("room", my_room);
#endif
#ifdef TM_LOAD_DEVICE_BME280
my_sensor_bme280.init();
#endif
#ifdef TM_LOAD_DEVICE_BME680
my_sensor_bme680.init();
#endif
#ifdef TM_LOAD_DEVICE_BH1750
my_sensor_illuminance.init();
#endif
#ifdef TM_LOAD_DEVICE_MHZ19
my_sensor_CO2.init();
#endif
#if defined(TM_LOAD_DEVICE_OLED_128X32) || defined(TM_LOAD_DEVICE_OLED_128X64)
my_display_oled.init();
my_display_oled.setBarChartRange(value_min_CO2, value_max_CO2); // = my_display_oled.setValueRange(value_min_CO2, value_max_CO2);
#endif
#ifdef TM_LOAD_DEVICE_LED_RING
my_display_led_rbg_ring.init();
my_display_led_rbg_ring.my_pixels.setBrightness(8);
my_display_led_rbg_ring.setValueRange(value_min_CO2, value_max_CO2);
#endif
#ifdef TM_LOAD_DEVICE_LED_KY_016
my_display_led_rbg_single.init();
my_display_led_rbg_single.setValueRange(value_min_CO2, value_max_CO2);
#endif
#ifdef TM_LOAD_DEVICE_4_DIGIT
my_display_4digit.init();
my_display_4digit.setValueRange(value_min_CO2, value_max_CO2);
my_display_4digit.displayValue(9000); // Displaying a value of 9000 for debugging purposes
#if defined(TM_DISPLAY_TIME)
my_display_4digit_loop_delay += 1;
#endif
#if defined(TM_LOAD_DEVICE_MHZ19)
my_display_4digit_loop_delay += 2; // twice the time than for the other values
#endif
#if defined(TM_LOAD_DEVICE_BME280)
my_display_4digit_loop_delay += 2; // 2 values
#endif
my_display_4digit_loop_delay = mySleep / my_display_4digit_loop_delay; // overwrite the counter by the resulting value in ms
// delay(1000);
#endif
} // end setup
//
//
// III. Loop
//
//
void loop()
{
timeStart = millis();
data_to_display = 8000 + loopNum; // dummy in case we have no sensor
display_shall_sleep = false; // set to on by default in each loop
//
// Influx: prepare empty dataset
//
#ifdef TM_LOAD_DEVICE_INFLUXDB
influx_data_set.clearFields();
#endif
//
// sync time every 7 days, but only at loopNum=0 for reducing connection problem timeouts
//
#ifdef TM_LOAD_DEVICE_INFLUXDB
#if defined(TM_DISPLAY_TIME) || (defined(TM_HOUR_SLEEP) && defined(TM_HOUR_WAKE))
if (loopNum == 0 && (getTimestamp() > timestampLastTimeSync + 24 * 3600 * 7))
{
my_influx.sync_time();
timestampLastTimeSync = getTimestamp();
}
#endif
#endif
//
// disable display if this is configured
// we only know the time if clock is initially set via WiFi and NTP
//
#ifdef TM_LOAD_DEVICE_INFLUXDB
#if defined(TM_HOUR_SLEEP) && defined(TM_HOUR_WAKE)
// check if we are at sleeping time
if (display_shall_sleep == false)
{
hour = getHour();
if (hour < TM_HOUR_WAKE || hour >= TM_HOUR_SLEEP)
display_shall_sleep = true;
}
#endif
#endif
//
// 1. read sensors
//
// 1.1 BME280: T, H, P
#ifdef TM_LOAD_DEVICE_BME280
data_bme280 = my_sensor_bme280.read_values();
// 0 = Temp, 1 = Humidity, 2 = Pressure
#ifdef TM_LOAD_DEVICE_INFLUXDB
if (data_bme280[0] > -100) // Temp must be larger than -100°C, else -> discard
{
influx_data_set.addField("temperature", data_bme280[0]);
influx_data_set.addField("humidity", data_bme280[1]);
influx_data_set.addField("pressure", data_bme280[2]);
}
#endif
#endif
// 1.2 BME680: T, H, P, IAQ, ...
#ifdef TM_LOAD_DEVICE_BME680
data_bme680 = my_sensor_bme680.read();
// data[0] = iaqSensor.rawTemperature;
// data[1] = iaqSensor.temperature;
// data[2] = iaqSensor.rawHumidity;
// data[3] = iaqSensor.humidity;
// data[4] = iaqSensor.pressure;
// data[5] = iaqSensor.gasResistance;
// data[6] = iaqSensor.iaq;
// data[7] = iaqSensor.iaqAccuracy;
// data[8] = iaqSensor.staticIaq;
// data[9] = iaqSensor.co2Equivalent;
// data[10] = iaqSensor.breathVocEquivalent;
#ifdef TM_LOAD_DEVICE_INFLUXDB
if (data_bme680[1] > -100) // Temp must be larger than -100°C, else -> discard
{
influx_data_set.addField("temperature", data_bme680[1]);
influx_data_set.addField("humidity", data_bme680[3]);
influx_data_set.addField("pressure", data_bme680[4]);
influx_data_set.addField("bme680_gasResistance", data_bme680[5]);
influx_data_set.addField("bme680_iaq", data_bme680[6]);
influx_data_set.addField("bme680_iaqAccuracy", data_bme680[7]);
influx_data_set.addField("bme680_staticIaq", data_bme680[8]);
influx_data_set.addField("bme680_co2Equivalent", data_bme680[9]);
influx_data_set.addField("bme680_breathVocEquivalent", data_bme680[10]);
}
#endif
#endif
// 1.3 BH1750: Illuminance
#ifdef TM_LOAD_DEVICE_BH1750
data_lux = my_sensor_illuminance.read();
#ifdef TM_LOAD_DEVICE_INFLUXDB
influx_data_set.addField("illuminance", data_lux);
#endif
if (display_shall_sleep == false && data_lux <= 1)
display_shall_sleep = true;
#endif
// 1.4 MHZ19: CO2
#ifdef TM_LOAD_DEVICE_MHZ19
if (millis() < 3 * 1000) // start is not reliable, so ignored
{
if (myVerbose)
{
Serial.println("not reading CO2 yet");
}
}
else // normal mode
{
data_CO2 = my_sensor_CO2.read_values();
// special handling for values of 0 and 380pmm
// MH-Z19 sometimes is interrupted by WiFi/InfluxDB or missing serial connection, resulting in always returning the same values,
// Workaround V1: retrying after a random sleep
if (data_CO2 == 0 || data_CO2 == 380)
{
if (myVerbose)
{
Serial.println("0 or 380ppm -> re-reading");
}
delay(random(100, 500));
data_CO2 = my_sensor_CO2.read_values();
}
// Workaround V2: rebooting after 10x the same CO2 values of 0 or 380ppm
if (data_CO2 == 0 || data_CO2 == 380)
{
count_same_CO2_values++;
if (count_same_CO2_values >= 10)
{
Serial.println("restarting");
my_esp32.restart();
}
}
else
{
count_same_CO2_values = 0;
}
data_to_display = data_CO2;
}
#ifdef TM_LOAD_DEVICE_INFLUXDB
influx_data_set.addField("CO2", data_CO2);
#endif
#endif
//
// 2. Displays
//
//
// 2.1 4-digit display
//
// 2.1.1 init
#ifdef TM_LOAD_DEVICE_4_DIGIT
if (display_shall_sleep == true)
{
my_display_4digit.ensure_sleep();
}
else
{
my_display_4digit.ensure_wake();
// 2.1.2a CO2 only
#if defined(TM_LOAD_DEVICE_MHZ19) && !defined(TM_LOAD_DEVICE_BME280)
my_display_4digit.displayValueAndSetBrightness(data_CO2);
delay(2 * my_display_4digit_loop_delay); // twice the time than the other values
#endif
// 2.1.2b H, T only
#if !defined(TM_LOAD_DEVICE_MHZ19) && defined(TM_LOAD_DEVICE_BME280)
my_display_4digit.setBrightness(1);
my_display_4digit.displayValue2p1(data_bme280[1]); // H
delay(my_display_4digit_loop_delay);
my_display_4digit.displayValue2p1(data_bme280[0]); // T
delay(my_display_4digit_loop_delay);
#endif
// 2.1.2c CO2, H, T
#if defined(TM_LOAD_DEVICE_MHZ19) && defined(TM_LOAD_DEVICE_BME280)
// my_display_4digit.setBrightness(0);
// for not on use same brightness for H and T as for CO2
my_display_4digit.displayValueAndSetBrightness(data_CO2);
delay(2 * my_display_4digit_loop_delay); // twice the time than the other values
my_display_4digit.displayValue2p1(data_bme280[1]); // H
delay(my_display_4digit_loop_delay);
my_display_4digit.displayValue2p1(data_bme280[0]); // T
delay(my_display_4digit_loop_delay);
#endif
}
#endif
// 2.1.3 time
#if defined(TM_LOAD_DEVICE_INFLUXDB)
#if defined(TM_DISPLAY_TIME) || defined(TM_HOUR_SLEEP) && defined(TM_HOUR_WAKE)
hour = getHour();
minute = getMinute();
my_display_4digit.displayTime(hour, minute);
delay(my_display_4digit_loop_delay);
#endif
#endif
//
// 2.2 LEDs
//
#ifdef TM_LOAD_DEVICE_LED_KY_016
if (display_shall_sleep == true)
my_display_led_rbg_single.setColor(0, 0, 0, 0);
else
my_display_led_rbg_single.displayValue(data_CO2);
#endif
#ifdef TM_LOAD_DEVICE_LED_RING
if (display_shall_sleep == true)
my_display_led_rbg_ring.displayValue(0);
else
my_display_led_rbg_ring.displayValue(data_CO2);
#endif
//
// 2.3 OLED Bar Chart
//
#if defined(TM_LOAD_DEVICE_OLED_128X32) || defined(TM_LOAD_DEVICE_OLED_128X64)
if (display_shall_sleep == true)
{
my_display_oled.ensure_sleep();
my_display_oled.appendValueToBarChart(data_CO2);
}
else
{
my_display_oled.ensure_wake();
my_display_oled.drawBarChart(data_CO2);
}
#endif
//
// 3. send data to InfluxDB
//
#ifdef TM_LOAD_DEVICE_INFLUXDB
my_influx.send_point(influx_data_set);
#endif
//
// 4. End of Loop
//
if (loopNum >= 31) // must be a multiple of 4 minus 1
loopNum = 0;
else
loopNum++;
sleep_exact_time(timeStart, mySleep);
} // end loop
//
//
// IV. Helpers
//
//
void sleep_exact_time(const unsigned long timeStart, const unsigned long duration)
{
const unsigned long timeEnd = millis();
// calc delay time based on start and stop
if (timeEnd < timeStart)
{ // overrun of millis() happened
delay(duration);
}
else if (timeEnd > timeStart + duration)
{ // loop took too long -> no sleep
}
else
{ // normal case
delay(duration - (timeEnd - timeStart));
// usually 0.1-0.2sec for one loop of reading my_sensor_bme280 and my_sensor_CO2 and pushing to InfluxDB
}
}
#ifdef TM_LOAD_DEVICE_INFLUXDB
#if defined(TM_DISPLAY_TIME) || defined(TM_HOUR_SLEEP) && defined(TM_HOUR_WAKE)
#include "time.h"
byte getHour()
// returns hour (00..23), if unknown returns 12
{
struct tm timeinfo;
if (!getLocalTime(&timeinfo))
{
Serial.println("Failed to obtain time, returning 12");
return 12; //-1;
}
return timeinfo.tm_hour;
}
byte getMinute()
// returns minutes (00..59), if unknown returns 0
{
struct tm timeinfo;
if (!getLocalTime(&timeinfo))
{
Serial.println("Failed to obtain time, returning 00");
return 0; //-1;
}
return timeinfo.tm_min;
}
unsigned long getTimestamp()
// returns unix timestamp, if not successful returns 0
{
time_t now;
struct tm timeinfo;
if (!getLocalTime(&timeinfo))
{
Serial.println("Failed to obtain timestamp, returning 0");
return (0);
}
time(&now);
return now;
}
#endif
#endif