working with circuit python version 8.2.10 — simple remote led control via http

Raspberry Pi Pico webserver via iPhone’s Safari web browser

Monday 18 March, Adafruit released version 9 of CircuitPython. I had been working with version 9 pre-releases up to release candidate 1, but due to some problems with RC1 I dropped back to 8.2.10. I might try version 9 on another different board I have, but for now I’m sticking with 8.2.10 until such time as I actually have to migrate to version 9 in the future.

The code I’m presenting came from an Adafruit project over a year ago, and I can’t recall where it was, but it was complicated with additional device control. I’ve taken the code and tidied it up a bit, especially the embedded HTML that’s used to render the webpage on a smartphone’s web browser. I’ve tested it on both Android as well as iOS, and it works fine regardless.

Note that I’ve also used the new CircuitPython’s feature of settings.toml to define not just the local SSID and password, but the static IP address of the CircuitPython webserver as well as if it’s running in debug mode or not. It’s a lot easier to change a small toml file than to go and try to change something in the code body.

Once up and running all you need to do is either turn on the on-board LED or turn it off. If you have a serial application attached to the USB port, then you’ll see debug information streaming out the port. This is because DEBUG on line 26 has been set to True in the settings.toml file.

And one more thing: in debug mode, you navigate to [STATIC_IP_ADDRESS]:5000 to see the page. When not in debug mode, then the webserver is on regular port 80.

import osimport timeimport ipaddressimport wifiimport socketpoolimport boardimport microcontrollerfrom digitalio import DigitalInOut, Directionfrom adafruit_httpserver import Server, Request, Response, POST#  On-device LED setupled = DigitalInOut(board.LED)led.direction = Direction.OUTPUTled.value = Falseipv4 =  ipaddress.IPv4Address(os.getenv('STATIC_IP'))netmask =  ipaddress.IPv4Address("255.255.255.0")gateway =  ipaddress.IPv4Address(os.getenv('GATEWAY'))wifi.radio.set_ipv4_address(ipv4=ipv4,netmask=netmask,gateway=gateway)wifi.radio.connect(os.getenv('AP_SSID'), os.getenv('AP_PASSWORD'))print(" Connected to WiFi access point...")pool = socketpool.SocketPool(wifi.radio)server = Server(pool, "/static", debug=os.getenv('DEBUG'))font_family = "sans-serif"# The HTML page body setup as an 'f' string.## Use {{ and }} if something within HTML *actually* needs to be in brackets# such as CSS style formatting.#def webpage():html = f"""<!DOCTYPE html><html><head><meta http-equiv="Content-type" content="text/html;charset=utf-8"><meta name="viewport" content="width=device-width, initial-scale=1"><style>html{{font-family: {font_family};background-color: #FFFFFF;display:inline-block;margin: 0px auto;text-align: center;}}h1{{color:#D35F8D;word-wrap: break-word;font-size: 35px;}}p{{font-size: 1.5rem;word-wrap: break-word;}}.button-red{{font-family: {font_family};display: inline-block;width: 99%;background-color: #DC143C;border: none;border-radius: 4px;color: white;padding: 16px 40px;text-decoration: none;font-size: 30px;margin: 2px;cursor: pointer;}}.button-green{{font-family: {font_family};display: inline-block;width: 99%;background-color: #228B22;border: none;border-radius: 4px;color: white;padding: 16px 40px;text-decoration: none;font-size: 30px;margin: 2px;cursor: pointer;}}p.dotted {{margin: auto;width: 90%;font-size:25px;text-align: center;}}</style></head><body><title>Pico W LED Control</title><h1>Pico W LED Control</h1><form accept-charset="utf-8" method="POST"><button class="button-green" name="LED ON" value="ON" type="submit">LED ON</button></a><button class="button-red" name="LED OFF" value="OFF" type="submit">LED OFF</button></a></form></body></html>"""return html#  route default static IP@server.route("/")def base(request: Request):  # pylint: disable=unused-argumentreturn Response(request, f"{webpage()}", content_type='text/html')#  Determine which button was pressed and execute associated action.#@server.route("/", POST)def buttonpress(request: Request):raw_text = request.raw_request.decode("utf8")print(raw_text)## If the led on button was pressed...if "ON" in raw_text:led.value = True## If the led off button was pressed...if "OFF" in raw_text:led.value = Falsereturn Response(request, f"{webpage()}", content_type='text/html')print(" Starting HTTP server...")try:server.start(str(wifi.radio.ipv4_address))print(f" Listening on http://{wifi.radio.ipv4_address}")##  If the server fails to start then kick-start the Pico W.#except OSError:time.sleep(5)print(" !!! OSERROR RESTARTING !!!")microcontroller.reset()print(" Start polling...")is_healthy = True;## DO NOT hard code as True, instead provide an exit if something goes wrong.#while is_healthy:try:# Poll for incoming requests.server.poll()# pylint: disable=broad-exceptexcept Exception as e:print(e)is_healthy = Falsecontinue
AP_SSID = "YOUR_SSID"AP_PASSWORD = "YOUR_SSID_PASSWORD"STATIC_IP = "192.168.0.10"GATEWAY = "192.168.0.1"DEBUG = "True"

software updates

Micropython 1.22 built with ESP IDF 5.2.1 via iPhone Safari

First, the big news. The latest updates to the Micropython code base now allow it to be built with Espressif’s ESP-IDF version 5.2. At the same time these changes were pushed out into the Micropython repository, Espressif dropped ESP-IDF 5.2.1, which I installed on my development system, then used to build my updated Micropython with the latest changes noted above. Micropython successfully built and I was then able to flash one of my ESP32-S3 development boards, and away I went with my stand-alone WiFi test board.

C++ application built with ESP IDF 5.2.1 via iPhone Safari

I then used the same ESP-IDF tool chain to build one of my C++ embedded applications, in which it joins my home WiFi access point and allows me to then bring up its simple web page. Again, everything built and flashed. This is with an ESP32-S3 development board with 32 MiB flash and 8 MiB external ram.

CircuitPython Update

One other update comes from CircuitPython: they’ve just dropped version 9.0.0 release candidate 0. I’ve downloaded and installed it on two of my boards that can handle it, and so far it’s doing just fine. If you like CircuitPython and use it for ESP32-S3-based boards, then by all means consider downloading and installing this version. It’s clean and stable enough for you to work with. Quoting from the release note, the key features out of many that caught my eye are:

  • Merge updates from MicroPython v1.19.1, v1.20.0, and v1.21.0.
  • Espressif: update to ESP-IDF v5.1.3.

Micropython and CircuitPython are moving rapidly, adding support for the latest boards as well as adding new functionality for existing devices, especially on the Espressif side of things.