Skip to content

sigmdel/xiao_esp32c3_wifi_switch

Repository files navigation

xiao_esp32c3_wifi_switch

XIAO ESP32C3 based Wi-Fi Switch (Version 0.0.8)

Source code that accompanies A Wi-Fi Switch for Domoticz using a XIAO ESP32C3:

Overview of the Wi-Fi Switch

The goal is to try to reproduce the basic functionality of Theo Arends' powerful Tasmota firmware which is used in many home automation devices. This is very ambitious, and consequently a step-by-step approach will be adopted. The end result will not be equivalent to Tasmota by any means but hopefully something will have been learned.

The image below shows the interaction between the microcontroller board, a XIAO ESP32C3, with an attached push button, a light level sensor, and a temperature and humidity sensor. As can be seen, it is connected to web browsers and a Domoticz home automation server.

Landscape

It's not just a matter of toggling a LED on and off with a button on a Web page; numerous examples can be found on the Web. Whenever the state of the light is changed locally with the button, the light's status has to be updated on the Web page displayed by all clients connected to the Web server and in the home automation system. Similarly, if the toggle button on a client's Web page is clicked, then the hardware controlling the light must be activated accordingly and the light's status must be updated in the home automation system and on all connected clients' Web page simultaneously. Likewise, if the virtual light switch in the home automation system is toggled on or off, the actual relay on the Wi-Fi switch must be updated and the new status of the light must be shown on all connected clients' Web pages.

Projects

All projects are independent and self-contained.

Private copies of the needed libraries are provided in the shared libraries directory, so it should be possible to compile all the projects without installing any library.

All projects will compile and run in the PlatformIO and Arduino IDEs as long as the ESP32 platform in installed and the correct board is selected. In the Arduino IDE it will be necessary to specify the Sketchbook location in the IDE Preferences in order to use the private copies of the libraries. Details are in section 6 of A Wi-Fi Switch for Domoticz using a XIAO ESP32C3 - Part 1.

Impatient?

Just want to test-drive the latest version? Follow these steps:

  1. Download the xiao_esp32c3_wifi_switch.zip archive from the current release

  2. Extract the xiao_esp32c3_wifi_switch-master directory on a drive. Let's call the full path to this directory project_dir.

  3. Copy the appropriate hardware driver, hardware.cpp from project_dir/xiao_esp32c3/hdw_xxx/hardware.cpp -where 'xxx is alt, kit, or mock) to project_dir/xiao_esp32c3/wifi_switch/hardware.cpp.

  4. Edit the pin definitions in project_dir/xiao_esp32c3/wifi_switch/hardware.cpp if necessary.

  5. Edit the default values in project_dir/xiao_esp32c3/wifi_switch/user_config.h_template and save as project_dir/xiao_esp32c3/wifi_switch/user_config.h.

  6. If using the Arduino IDE:

    1. set Preferences' Sketchbook location to project_dir/xiao_esp32c3/.
    2. open the project_dir/xiao_esp32c3/wifi_switch/wifi_switch.ino sketch.
    3. make sure the ESP32 by Espressif boards definition is installed with the Boards manager.
    4. select the XIAO_ESP32C3 board.
    5. select the appropriate port.
    6. compile and upload the sketch.
    7. open the serial monitor and view the log messages to find the IP address of the XIAO.
  7. If using PlatformIO:

    1. open the project_dir/xiao_esp32c3/ project.
    2. compile and upload the project.

Good luck.

Opening a project in PlatformIO

Open a PlatformIO project by clicking on the Quick Access Open Project button in the Home page of PlatformIO. Then navigate to the project directory (such as 01_simplified_hdw_version) which contains the project configuration file platformio.ini and click on the `Open "<project name>".

Opening a sketch in Arduino

Open an Arduino sketch by clicking on the File/Open menu. Then navigate to the sketch file (such as 01_simplified_hdw_version\simple_wifi_switch\simple_wifi_switch.ino) and then click on the Open button in the file open dialog window.

01_simplified_hdw_version

In this first iteration of the ESP32-C3 firmware, a hardware abstraction layer is built to handle

  • a normally open push button,
  • a DHT20 temperature and humidity sensor connected over the I²C serial bus,
  • a light sensor based on the LS06-S.

A future hardware layer 03_alt_wifi_switch includes support for more types of sensors.

The asynchronous web server is capable of handling multiple clients at the same time. Each Web client reloads the Web page every five seconds because the HTML page served has a refresh meta tag.

This project corresponds to sections 2 to 5 of A Wi-Fi Switch for Domoticz using a XIAO ESP32C3 - Part 1.

02_basic_wifi_switch

Integration of the Wi-Fi switch into the home automation system Domoticz is added in this second iteration of the ESP32-C3 firmware. Communication with the Domoticz server is done with HTTP requests.

Domoticz can be used to easily modify the behaviour of the Wi-Fi switch so that it becomes a timed light which stays off after a given period of time or an auxiliary light that gets turned on and off whenever another Domoticz controlled switch is turned on or off. It is also very easy to transform the Wi-Fi switch into a night light that turns on at dusk and back off at sunrise in Domoticz. The ../lua directory contains a dzVents Lua script that turns the Wi-Fi switch on or off based on the light level measured by the on-board sensor.

The ../lua directory contains a dzVents Lua script that automatically adjusts the humidity status displayed for all Domoticz supported devices that measure temperature and humidity and optionally barometric pressure.

This project corresponds to sections 7 and 8 of A Wi-Fi Switch for Domoticz using a XIAO ESP32C3 - Part 1.

03_alt_wifi_switch

This third project is project 2 but with a different hardware abstraction layer. It can handle three types of temperature and humidity sensors (DHT11, DHT20 or DHT22) and two types of light level sensors (LDR or LS06-S photodiode). Additionally, a temperature and humidity sensor can be emulated and the same is possible with the light level sensor. Finally, a rolling average of the light level measurements can be enabled to smooth out fluctuations caused by clouds or other passing shadows.

This project corresponds to section 10 of A Wi-Fi Switch for Domoticz using a XIAO ESP32C3 - Part 1.

04_ajax_update

Full Web page reloading using a refresh meta tag is replaced by asynchronous JavaScript and XML (AJAX) based Web page updates.

This project is described in section 1 of A Wi-Fi Switch for Domoticz using a XIAO ESP32C3 - Part 2 (Asynchronious Web Page Updates).

05_websocket_update

Full Web page reloading using a refresh meta tag is replaced by asynchronous websocket based Web page updates.

This project is described in section 2 of A Wi-Fi Switch for Domoticz using a XIAO ESP32C3 - Part 2 (Asynchronious Web Page Updates).

06_sse_update

Full Web page reloading using a refresh meta tag is replaced by asynchronous Server-Sent Events (SSE) based Web page updates.

This project is described in section 3 of A Wi-Fi Switch for Domoticz using a XIAO ESP32C3 - Part 2 (Asynchronious Web Page Updates)

About the last 3 xxx_update projects

They present 3 techniques that could be used to update the information displayed on the Web page without reloading the page itself as done in the first three projects. While Tasmota uses AJAX, and Websockets probably represent the most powerful technique, Server-Sent Events and AJAX will be used starting in 08_ticker_hdw.

07_with_log

Carrying on from 06_sse_update, this version adds a private logging facility. It is implemented as a FIFO queue with replacement of older entries when adding a log message when the queue is already full. That way, it is possible to log messages in an interrupt service routine and even before the serial port is up. Actually sending log messages is done safely in the loop() thread. Started to remove blocking operations. There is no longer any waiting for the WiFi connection in setup(). Similarly failed initialization of the temperature sensor will no longer block the execution of the firmware.

There is still a problem with HTTP requests to a missing Domoticz server or to an incorrect address. Request will still block until a timeout is reached. Currently, the timeout is set at the lowest possible value (1 second), so firmware will not respond to the button during that time. The solution would seem to be to find an asynchronous HTTP request library or to make each request in a separate thread or task.

08_ticker_hdw

No luck so far with asynchronous HTTP request libraries. Tests of embedding a blocking HTTP request in a (RTOS) task are positive. However, if tasks are used for this, they should be used for everything else including button presses. Consequently, another approach is used: HTTP requests are buffered in a FIFO queue with replacement. This is the same technique as used with the log. To see how that works, ensure that the serial log level is set to LOG_DEBUG (recompile and upload the firmware if the level had to be changed), stop Domoticz, and press the button many times in quick succession. Quickly, the buffer will be filled and old requests will be dropped. Nevertheless, the LED will follow the state of the button because the hardware is now polled with a Ticker. Every 25 milliseconds (POLL_TIME is set in hardware.h), a hardware timer interrupt takes care of checking the state of the button and the sensors if need be. Since this is an interrupt, the button state is monitored even as an HTTP request might be in progress.

The alternate hardware drivers, first shown in 03_alt_wifi_switch have been updated and are contained in hdw_alt/hardware.cpp. This is a drop-in replacement for ticker_hdw/hardware.cpp. Note that now change is requires to the header file hardware.h.

09_with_cmd

This version continues segregating all configuration values that can be modified by the user at run-time into config.h and config.cpp. As before the default values are in user_config.h which is not included in the repository. Use user_config.h.template as a model.

The important addition in this version is a command interpreter in commands.hpp and commands.cpp. Currently commands, which for the most part consist of viewing or changing the configuration at run-time, can be entered in the serial console or in the Web interface console. There will not be a Backlog command à la Tasmota. Instead, multiple commands can be specified at once as long as they are separated by a ";".

00:02:53.737 CMD/inf: Command from webc: help config; config; idx; version
00:02:53.737 CMD/inf: config [load|default|save [force]] | [auto (off|on)]
00:02:53.738 CMD/inf: Config version: 1, size: 548, auto(save): on
00:02:53.738 CMD/inf: Domoticz virtual Idx: switch = 1, light sensor = 3, temp + humid sensor = 2
00:02:53.738 CMD/inf: Firmware version 0.0.5 (Apr 26 2023 21:51:48)

This first version of the command interpreter is just a proof of concept and it will not be truly operational until the configuration can be saved to non-volatile memory and read back from it when the ESP32C3 boots. Right now, that seems the simplest way of a change from dynamic IP address to a static IP address.

10_with_config

The user-defined configuration can be saved to or loaded from non-volatile storage (NVS). With improvements and a revamped ESP restart command, the command interpreter is now working properly. In version 0.0.6, the new unit wifiutils which takes care of the network details has been simplified.

Network configuration is handled with two commands: a new wifi command used to specify the Wi-Fi access point and the staip command to choose between getting a dynamic IP address from the network DHCP server or using a static IP address. Changes to the configuration with these two commands have no effect until the device is restarted. There is no reason to keep secrets.h and secrets.h.template since network credentials are now included in the configuration .

The list of command and their syntax can be found in A Wi-Fi Switch for Domoticz using a XIAO ESP32C3 - Part 4: Commands. This is still very much a work in progress, so commands will be added and changes to existing commands may be made. Also, the command interpreter needs much reworking. It's as if each command is parsed with a different approach in an attempt to find a common approach that could be used in all cases.

Added the AsyncElegantOTA library by Ayush Sharma. It is not how I usually handle over-the-air updates, but it was very simple to add and does handle authentication if that is required.

11_with_wm

A Wi-Fi manager is added. Well, that's a bit pretentious; only an alternate root Web is provided. When the Wi-Fi switch has lost the connection to the wireless network for more than a specified time interval (5 minutes as defined in the default user configuration), it starts an access point. It will be necessary to log onto that new Wi-Fi network, named KITCHENLIGHT-AP with 12345678 as a password, to open the Web server to get access to a form used to specify the Wi-Fi credentials and from there try to connect to the Wi-Fi network. As soon as a Wi-Fi connection is established, the access point is brought down.

This required adding two new commands

  • ap to manage the access point name and password
  • apip to manage the access point IP address and subnet mask

and modifying the time command by inserting the ap parameter to set the disconnection time interval before starting the access point.

The corresponding fields had to be added to the configuration structure config_t. Also, three Web pages were added in html.h not just one and two functions were appended to wifiutils to start and stop the access point. Starting and stopping the access point is done in the WiFiModule in main.cpp.

When connected to the wireless network started by the Wi-Fi switch, it is possible to get access to the main Web page at 192.168.4.1/index.html (or whatever the AP's IP address is). That way, the relay can be turned on or off, the sensor values will be updated and the firmware can be updated. The Wi-Fi switch configuration can also be modified with the Web console. If one prefers to have a completely separate Wi-Fi manager, then look at webserver_xl/webserver.cpp.

For some reason, it can be difficult to connect my desktop computer (Linux Mint 20.1 with a 5.15.0-72-generic kernel) to the access point. Sometimes, turning the radio off, and then back on, can solve that problem.

$ sudo nmcli radio wifi off
$ sudo lshw -C network       
$ sudo nmcli radio wifi on

The lshw is used to verify that the interface is actually off. When this trick does not seem to work even after a number of tries, I find that connecting to the home Wi-Fi network, waiting until an IP version 4 address is assigned to the computer Wi-Fi interface and then closing the connection seems to "prime" the wireless interface. If all this fails, then the desktop can be rebooted and it usually does not have problem connecting to the XIAO access point if it is the first wireless network used. That is obviously a tedious method that grows old very quickly. It's all very frustrating but, interestingly, that problem is not encountered when I use Android devices.

The simulated sensors in the alternative hardware driver in 08_ticker_hdw/hdw_alt/hardware.cpp has been extracted and moved to a separate file. There are now three different hardware driver:

  • hdw_kit/hardware.cpp that uses the SeeedStudio Start Kit (used by default)
  • hdw_alt/hardware.cpp that uses generic sensors
  • hdw_mock/hardware.cpp that emulates sensors

Just copy the desired driver over the with_mw/hardware.cpp.

11_with_mqtt

This latest (and last?) version adds MQTT functionality. Most of the work is done in the mqtt module, but changes to other modules were made. The domoticz module now communicates with the Domoticz server using MQTT messages. If it cannot connect to the MQTT server, it will update the Domoticz database using HTTP requests as before. Along the same line, it is not necessary to remove On and Off actions of the virtual light device that used HTTP requests to have the Wi-Fi switch turn on or off the relay. This can be seen as a backup of the MQTT channel. The MQTT log facility was added to logging. Four MQTT topics were added as user-settable values in config. Two are the Domoitcz in and out topics, a third is the topic used for the log and the fourth is the subscription topic for commands. All commands that can be entered in the Web console or through the serial interface can be sent as the payload of an mqtt message.

The access point is entirely managed within the wifiutils unit.

The espRestart function in main.cpp has been revamped. Currently 8 restart modes are defined. This is much too granular and will probably be reduced in the future after some testing. Currently, long button presses can initiate one of three restart modes.

time press (seconds) restart level actions before actual restart
greater than 30 7 restore the default config, then sets dynamic IP, clears wifi credentials, saves config
between 10 and 30 3 sets dynamic IP, saves the configuration
between 0.75 and 10 0 saves the configuration if it has been changed

Preliminary tests show that the source code can generate working firmware for ESP32-S2 based devices. That is another single core microcontroller. It uses the Xtensa LX7 microprocessor which makes the ESP32-S2 closer to the ESP8266 than the ESP32-C3. The platformio.ini shows the settings used for the AI-Thinker nodeMCU ESP-32H dev kit which is rather more finicky than the XIAO ESP32C3.

License

Copyright 2023, Michel Deslierres. No rights reserved.

While the copyright pertaining to included libraries by others must be respected, all the source code in this repository created by Michel Deslierres is in the public domain. In those jurisdictions where this may be a problem the BSD Zero Clause License applies.