A temperature and humidity sensor using a DHT22/11 device and UDP to transmit the data to a server.
- History
- Overview
- Details
- Future Modifications
Table of contents generated with markdown-toc
I had been tinkering with using an ESP8266 and a temperature/humidity sensor to record data within my home. I hadn't really moved forward with it until the end of December 2017. And what really prompted me to move forward with the project was the brutally cold temperatures we experienced in Chicago. I was concerned about water pipes freezing, or my furnace breaking down. So the completion of this project (sensors, server, database, and web client) became very important. Fortunately no pipes froze, and my furnace continued working! I was luckier than some of my neighbors!
Each sensor device consists of an ESP-01S and a DHT22 temperature/humidity sensor. However most any ESP8266 platform should work.
The operation of the devices are configurable and utilize UDP to communicate status and data to a server implemented to run on NodeJS.
This application requires a server capable of receiving UDP packets. I've chosen NodeJS for this, and a server can be found in my node-dht-udp repository.
It is the responsibility of that server to listen for data from the sensors and forward it to a database and to any connected web clients.
The devices transmit two types of UDP traffic, multi-cast UDP and UDP. The multi-cast messages are intended for communicating device status to one or more listeners. The other messages are sent to a specific server and contain sensor data.
Here's an example of a typical data message -
- Device Sensor Data -
{"dev_id":"ESP_49ECF6","seq":2,"t":67.28,"h":26.20}
dev_id
- The ID of the device, for the ESP8266 devices this is typically the network ID of the ESP8266.seq
- This sequence number is incremented every time the DHT-XX is queried for data. There can be gaps in the sequence and it indicates that a reading has occurred but the amount of change was not sufficient to send a message. The server can use it to aid in interpolation of values between readings, and for smoothing out graphs.t
- The temperature in the scale (F or C) that was configured.h
- The relative humidity
Here are some examples of the status messages that a sensor device might send -
- Device successful start -
{"dev_id":"ESP_49ECF6","status":"APP_READY"}
- Device sensor error -
{"dev_id":"ESP_49ECF6","status":"SENSOR_ERROR","msg":"Too many NaN readings from sensor"}
- Device sensor recovery -
{"dev_id":"ESP_49ECF6","status":"SENSOR_RECOVER","msg":"Recovered after NaN from sensor"}
The primary purpose of the heartbeat is enhance the monitoring of the state of a sensor device. A heartbeat will occur when the temperature and humidity have not changed more than their delta for a period of time.
The period of time that will pass between heartbeats is the sensor read interval times 4. For example, if the DHT-XX device is read every 5 minutes a heartbeat will occur every 20 minutes unless a data message is sent. The transmission of a data message will reset the heartbeat interval counter.
There are several messages associated with the heartbeat -
- Heart Rate Announcement -
{"dev_id":"ESP_49ECF6","status":"HEART","msg":"heartbeats @ 5 min(300000ms)"}
- Only sent once during start up & initialization.
- Heartbeat Pulse -
{"dev_id":"ESP_49ECF6","status":"TICK" | "TOCK","msg":"beatcount = 5"}
- The
status
will alternate betweenTICK
andTOCK
each time the message is sent. Thebeatcount
value is a counter of how many heartbeats have occurred to that point. This is an optional message, and it is typically disabled. To enable it change the value ofesp8266-dht-udp.ino:sendbeat
totrue
.
- The
- Heartbeat Sensor Data -
{"dev_id":"ESP_49ECF6","seq":1539,"t":59.72,"h":35.70,"last":{"t":60.08,"h":36.20}}
- Sent when a heartbeat occurs. There are two sets of temperature & humidity values. The first is the current reading directly from the DHT-XX sensor. And the second, in the
last
object are the values that were sent in the last data message that was sent before the heartbeat.
- Sent when a heartbeat occurs. There are two sets of temperature & humidity values. The first is the current reading directly from the DHT-XX sensor. And the second, in the
NOTE : The heartbeat can be disabled by commenting out #define HEARTBEAT
in esp8266-dht-udp.ino
.
The configuration source code is based on my ESP8266-config-data-V2 repository. Therefore only the configurable items and their use will be described here.
The configuration files are kept in the data
folder.
Some configuration files may contain sensitive information that should not be placed into a public repository. In order to prevent them from getting into the repository their filenames begin with an underscore. This is accomplished with an entry in this repository's .gitignore
file. However there are example configuration files provided that to not have the underscore in their names.
This file is the first configuration file read by the application. It provides a location where the remaining configuration files are listed. It also contains the application name, and a boolean value for enabling or disabling run-time output to the serial port. Here are the contents of the sample appcfg.json
file -
{
"appname":"Your App Name Here",
"debugmute":false,
"wificonfig":"/wificfg.json",
"clientconfig":"/clientcfg.json",
"mcastconfig":"/multicfg.json",
"sensorconfig":"/sensorcfg.json"
}
To keep the contents of this file secure make a copy of it and prepend the underscore to its name.
The data/wificfg.json
file contains one or more WiFi SSIDs and passwords. Here are the contents of the sample wificfg.json
file -
{ "apoints":[
{"ssid":"your ssid-1 here","pass":"your wifi password-1 here"},
{"ssid":"your ssid-2 here","pass":"your wifi password-2 here"}
],"apcount":2}
The code responsible for connecting to an access point will multiple attempts. This is described in the README of the ESP8266-config-data-V2 repository.
To keep the contents of this file secure make a copy of it and prepend the underscore to its name. Be sure to edit your data/_appcfg.json
file to access the correct file.
The data/clientcfg.json
file contains one or more entries that each contain the IP address and port number of a UDP enabled server that the application can access. Here are the contents of the sample clientcfg.json
file -
{
"udp1":{"addr":"server IP address","port":54321},
"udp2":{"addr":"server IP address","port":54321}
}
The server is chosen in esp8266-udp.cpp
, initUDP()
. Edit this line -
if(c_cfgdat->getServer("udp1", udpClient)) success = true;
and change "udp1"
to a different string as needed.
The chosen configuration data is read and parsed into the esp8266-udp.cpp:udpClient
object.
To keep the contents of this file secure make a copy of it and prepend the underscore to its name. Be sure to edit your data/_appcfg.json
file to access the correct file.
The device has the ability to request the server's IP address and port number that it uses for UDP messages. It accomplishes this by broadcasting a REQ_IP
status message. When received by the server it will respond directly to the device with its UDP address information.
After the device receives the server's reply it will parse it into esp8266-udp.cpp:udpClient
and over write any values set when the data/clientcfg.json
file was read and parsed.
NOTE : This can be disabled by commenting out the line #define QUERY_SERVER
in esp8266-ino.cpp
.
The data/multicfg.json
file contains the IP address and port number for the intended multi-cast destination. Here are the contents of the file -
{
"addr":"224.0.0.1",
"port":54321
}
This file does not contain sensitive configuration data. So it is not necessary to prepend the underscore to its name.
The mimic feature is used for allowing for the replacement of a failed device with a new one, and having the new device "mimic" the hostname
and device ID of the one it replaced. This become useful when there is a large amount of data that's been collected(server side) and it's indexed on the device ID.
To enable mimic for a device:
-
create a file named
_device_id.json
, wheredevice_id
is the ID of the current device. NOTE: The file name must begin with an underscore "_"! For example_ESP_BEEFEE.json
. -
edit the file and add:
{
"mimic": "ESP_12AB34"
}
Where "ESP_12AB34"
is the hostname of the device that is being replaced.
-
save the file and upload to the device
-
when the sketch runs it will look for a json file where it's own id is in in the file name. If found it will take the ID from the file and replace it's current ID. It will use that ID when sending statuses and data.
Here are the contents of the sensorcfg.json
file -
{
"type":"DHT22",
"pin":"D6",
"scale":"F",
"interval":300000,
"error_interval":10000,
"report":"CHG",
"delta_t": 5,
"delta_h": 10
}
The data/sensorcfg.json
file contains the configure the application for one of two specific temperature/humdity sensors -
type
- Sensor type, either"DHT11"
or"DHT22"
. At this time these are the only sensors supported.pin
- EPS8266 pin number, this is the pin number of the ESP8266 that is used for communication with the DHT sensor.- NOTE : This pin setting is ignored if an ESP-01 is used. On that platform GPIO2 will be used instead and is not configurable. See
sensor-dht.cpp
and look forARDUINO_ESP8266_ESP01
for the associated code.
- NOTE : This pin setting is ignored if an ESP-01 is used. On that platform GPIO2 will be used instead and is not configurable. See
scale
- Temperature scale, this is used to select Fahrenheit or Celsius.interval
- Sensor reading interval, this is the duration in milliseconds between subsequent sensor data readings.error_interval
- Sensor retry interval, this is the duration in milliseconds between subsequent sensor data readings when an error (typically the sensor will return NaN) occurs.report
- Reporting type, the current choices are"ALL"
or"CHG"
. Their meanings are -"ALL"
- report the sensor data every time the sensor data is read."CHG"
- only report sensor data if the temperature or humidity values have changed.
delta_t
&delta_h
- If the reporting type is"CHG"
then this is the amount of required change before the temperature or humidity are reported. The integer value kept here is divided by 10 to create a floating point value. Then if the amount of change (temperature or humidity) is greater then the data is sent.
This file does not contain sensitive configuration data. So it is not necessary to prepend the underscore to its name.
I experimented with OTA with limited results. And the device would not appear reliably in the Arduino IDE. So it has been disabled. To enable OTA remove the comment on the line //#define USE_OTA
in esp8266-dht-udp.ino
.
The OTA configuration is located in data/otacfg.json
. At this time only the otadur
is used. It is the amount of time that the device will wait for OTA to begin before it starts the application.
schematic goes here
I usually don't like to wait very long for parts and components, so I normally use Amazon as my supplier. (The links below connect to Amazon)
- Elegoo 6PCS 170 tie-points Mini Breadboard kit for Arduino
- MagiDeal 10pcs Female MICRO USB to DIP 5-Pin Pinboard 2.54mm micro USB type
- ICstation AMS1117-3.3 DC Voltage Regulator Step Down Power Supply Module 4.75V-12V to 3.3V 800mA (Pack of 5)
- MakerSpot 6mm 2 Pin Panel PCB Momentary Tactile Tact Push Button Switch Through Hole Breadboard Friendly x 10 pack
- DIYmall ESP8266 ESP-01 ESP-01S Breakout Board Breadboard Adapter PCB for Serial Wifi Transceiver Network(pack of 5pcs)
- Makerfocus 4pcs ESP8266 ESP-01S WiFi Serial Transceiver Module with 1MB Flash for Arduino
- Sensor choices -
- Gowoops 2 PCS DHT22 Temperature Humidity Sensor Module Digital Measurement for Arduino Raspberry Pi 2 3
- HiLetgo 1pc DHT22/AM2302 Digital Temperature And Humidity Sensor Module Temperature Humidity Monitor Sensor Replace SHT11 SHT15 for Arduino Electronic Practice DIY
- WINGONEER 5PCS of Temperature Humidity Sensor Module Digital DHT11 for Arduino Raspberry Pi 2 3
- Power Supply -
My preference of sensors is the DTH22. It seems to have better accuracy and less drift than the DHT11.
The following modifications were made to the Adafruit DHT source files src/adafruit/DHT.h
and src/adafruit/DHT.cpp
-
- Added an overloaded contructor -
void DHT::DHT(void)
- Added an overloaded begin function -
void DHT::begin(uint8_t pin, uint8_t type)
The purpose of the changes was to allow the DHT class to be instantiated before the sensor configuration was read and parsed.
Add a "version" member to the appcfg.json
file. This would contain a string formatted as "0.0.0"
using the typical "major"."minor"."patch" scheme. For example -
{
"appname":"Your App Name Here",
"appver":"0.1.0",
"debugmute":false,
"wificonfig":"/wificfg.json",
"clientconfig":"/clientcfg.json",
"mcastconfig":"/multicfg.json",
"sensorconfig":"/sensorcfg.json"
}
In addition single letters such as "b
" could be added to the end of the version string.
To accommodate devices with configuration differences the following changes will be made -
- One or more configuration files will can be named using the ESP-01 device hostname. For example in
appcfg.json
-
{
"appname":"Your App Name Here",
"debugmute":false,
"wificonfig":"/wificfg.json",
"clientconfig":"/clientcfg.json",
"mcastconfig":"/multicfg.json",
"sensorconfig":"/sensorcfg-ESP_49F542.json"
}
Where "sensorconfig":"/sensorcfg-ESP_49F542.json"
would specifiy the sensor config file for a specific device. The addtion of ESP_49F542
would match the hostname of a specific device.
Commands could be issued from the server that would alter one or more configuration items. For example the following could be reconfigured -
- Sensor -
- scale
- interval
- report type
- delta
- TBD