A template for ESP8266 programming using VSC + PlatformIO supporting MQTT, OTA-flashing, ESP Deep-Sleep and VCC readouts.
A (local) MQTT broker is mandatory for OTA-Flashing. With deactivated OTA-flashing, you might remove MQTT functionality.
Additionally, personal settings like WIFI SSID and Passphrase will be taken from local environment variables, see platformio.ini
.
To be able to use DEEP_SLEEP functionality, you will most probably need a small hardware modification for you ESP board: connect pin D0 to RST pin. This will allow the ESP to wake up after the defined sleep time as defined in the include/generic-config.h
file. Note that there is a hardware limitiation on the ESP8266 of approx. 1 hour for maximum deep sleep time.
As the main intention of this program is to run on a MCU that is powered off (sleeping) most of the time, we will only work with retained messages here! This ensures that a client subscribing to a topic will receive the last value published "instantly" and does not need to wait for someone to publish "latest news".
ATTENTION: The current version requires retained messages for all topics you subscribe to! Not having retained message for a subscribed topic will lead to an endless loop until a message is being received. This behavior has been set up to speed up receiving messages for all subscribed topics after initiating the connection to the MQTT broker (to minimize sketch runtime / maximize battery lifetime).
In addition to the platformio.ini
file, see header files in the include/
folder for additional settings.
Configureable settings (include/*-config.h
) should be documented well enough there - hopefully wink.
In order to run OTA updates, you will need at least the following MQTT topics on your broker (case sensitive) to be pre-created with the default retained message so ESP can subscribe to them:
-
topic/tree/OTAupdate
- default retained Message: off
This will be translated to a bool variable in the sketch. You will need to set the topic value either to "on" or "off". During normal operation, this Topic needs to be set to "off". If you want to run an OTA-update on your ESP, set it to "on" (retained).
After a successful update, the ESP will reset this flag to "off". -
topic/tree/OTAinProgress
- default retained Message: off
This is a helper flag topic required by the ESP. -
topic/tree/OTAstatus
- default Message: none
The ESP will publish OTA status strings here. No need to pre-create this topic, sketch does not subscribe to it (only publish). -
topic/tree/Vcc
- default Message: none
The sketch will publish the voltage measured on the 3.3V supply here if enabled inplatformio.ini
. Note that the accuracy is quite low, but it is good enough to detect if the battery is running low when you supply the ESP in example by a LiFePo4 accumulator directly on the 3.3V pin.
If you want to improve accuracy, measure the actual voltage with a multimeter and adopt theVCCCORRDIV
in thehardware-setup.h
file.
Note that we do not subscribe to this topic, we only publish to it.
Note that the ClientName
configured in the platformio.ini
file will also be used as the hostname reported to your DHCP server when the ESP fetches an IP-address. This is especially important, as OTA-flashing will also require your networking environment to be able to resolve this hostname to the ESP's IP-address!
See upload_port
setting in the platformio.ini
file. If you're having troubles with OTA-flashing, you might want to check that first by pinging the configured ClientName
.
I will give a rough walkthrough on the first steps, assuming you have a working PlatformIO environment:
- Prepare system environment variables with WIFI and MQTT Data:
- WIFI_SSID
- WIFI_PSK
- MQTT_BROKER --> IP-Address of your broker
- OTA_PWD --> Passphrase to use for OTA updates
- Adopt
ClientName
inplatformio.ini
as needed - Prepare
platformio.ini
for wired flashing Deactivate OTA-flashing in the board specific area:
;upload_protocol = ${common_env_data.upload_protocol}
;upload_port = ${common_env_data.upload_port}
;upload_flags = ${common_env_data.upload_flags}
- Adopt board info (
platformio.ini
) and hardware settings ininclude/hardware-config.h
if needed - Compile and flash
To re-flash the sketch OTA:
- Prepare
platformio.ini
for OTA flashing Activate OTA-flashing in the board specific area:
upload_protocol = ${common_env_data.upload_protocol}
upload_port = ${common_env_data.upload_port}
upload_flags = ${common_env_data.upload_flags}
- Set topic
topic/tree/OTAupdate
retained "on" - wait for the ESP to start up (or reset your ESP)
- When the ESP boots, it will slowly blink the onboard LED until all MQTT topics are received
- After that, the onboard LED will start flashing rapidly. The ESP is now waiting for the new binary to be uploaded
- Click "Upload" in PIO to compile and upload the new sketch
- When the upload has finished, the ESP will boot the new sketch and finish the OTA update process by setting
topic/tree/OTAinProgress
andtopic/tree/OTAupdate
to "off". - You can verify the status by reading the
topic/tree/OTAstatus
topic, which should throw the string "update_success"
Your OTA flashing might fail with the following error:
Sending invitation to esp-test
Authenticating...OK
20:31:25 [INFO]: Waiting for device...
20:31:35 [ERROR]: No response from device
*** [upload] Error 1
If you run Windows, make sure that your local firewall (where you run VSC) allows C:\Users\your_username\.platformio\python3\python.exe
to freely communicate (Any protocols, any ports - inbound and outbound; use the full path as described, do not use %USERPROFILE%
environment variable, it won't work!).
To add your own functionality to the template, you will need to adopt the following files:
-
include/mqtt-ota-config.h
Update theTOPTREE
to your needs. -
include/user-config.h
Define/declare your topics along with required global vars and libs here. -
src/user_setup_loop.cpp
Add your desired functionality to theuser_loop
anduser_setup
functions. -
src/mqtt-subscriptions.cpp
Add an array element to theMqttSubscriptions
struct array, in example:
{.Topic = ota_topic, .Type = 0, .Subscribed = false, .MsgRcvd = 0, .BoolPtr = &OTAupdate }
The following data types are supported (parameter Type
):
-
BOOL (Type = 0) Use the
.BoolPtr
to point to a globalbool
variable where you want to store the received messages for this topic. The BOOL type expects either "on" or "off" text messages to be received from the subscribed topic. -
INT (Type = 1) Use the
.IntPtr
to point to a globalint
variable. The string function.toInt()
will be used to decode the message to an integer value. -
FLOAT (Type = 2) Use the
.FloatPtr
to point to a globalfloat
variable. The string function.toFloat()
will be used to decode the message to a float value. Make sure to use dots as decimal point in your messages.
I have tried to get rid of the delay()
s implemented to allow background WiFi processing (by using WiFiClient.flush
instead of delays and configuring WiFiClient.setNoDelay
), but i was not able to get satisfying MQTT communication without them. Either it took too long to fetch new messages from subscribed topics, or sending new messages crashed (resetted) the ESP.
My recommendation is, to make sure to add at least a 100ms delay after sending a bunch of MQTT messages. You may also want to use the MqttDelay
function if you add longer delays (above 200ms), as it will automatically call the PubSubClient loop
function to fetch new MQTT messages every 200ms of delay.
In addition, i have added a user_loop
runtime specific delay in the main loop in v1.1.0: if the user_loop
takes less than 100ms for execution, a 100ms delay will execute in the main loop.
ATTN: OTA flashing did not work due to an error in macro handling!
- Fixed error in macro handling
- Speedup receiving messages of subscribed topics
- Major code cleanup
- Moved user specific stuff into dedicated files / functions (
user_setup
anduser_loop
) - Added
MqttDelay
function which handles MQTT connection/subscriptions while delaying - README update on
ClientName
limitation
- Fixed
Clientname
limitation - Added ESP reboot when cancelling OTA Update
- Added
user_loop
runtime dependent 100ms delay in main loop - Reworked MQTT subscriptions
- OTA updating support now mandatory
user_loop
runtime dependent delay now configurable inplatformio.ini
- removed additional (unnecessary) delays
- Switched
MsgRcvd
from typebool
touint32_t
counter to be able to keep track if new messages arrive for subscribed topics (increases on new valid message arrival) - Minor improvement to allow re-defining MQTT topic tree in
user-config.h
to allow over-writingmqtt-ota-config.h
when upgrading the framework - Reading VCC now optional (configured via define in
platformio.ini
) JustBooted
global flag available;true
when running main loop for the first time, then set tofalse