ESP32 OTA update with Web Browser: custom web interface – 3

Spread the love

ESP32 is a powerful microcontroller that can be used for a wide range of Internet of Things (IoT) applications. One of the key features of ESP32 is its ability to receive Over-The-Air (OTA) updates. OTA updates allow you to remotely update your ESP32 firmware without having to physically connect it to a computer or USB cable.

In this tutorial, we will explore how to implement OTA updates on an ESP32 using a web browser with a custom interface. This means that we will be able to upload firmware updates to the ESP32 through a web page hosted on the device itself.

ESP32 OTA update with Web Browser: custom web interface
ESP32 OTA update with Web Browser: custom web interface

We will begin by setting up the necessary libraries and configuring the ESP32 to connect to a local network. We will then create a custom web interface using HTML and JavaScript, which will allow us to browse for and upload firmware files to the ESP32.

Here some common devices ESP32 Dev Kit v1 - TTGO T-Display 1.14 ESP32 - NodeMCU V3 V2 ESP8266 Lolin32 - NodeMCU ESP-32S - WeMos Lolin32 - WeMos Lolin32 mini - ESP32-CAM programmer - ESP32-CAM bundle - ESP32-WROOM-32 - ESP32-S

OTA (Over the Air) update is the process of uploading firmware to an ESP32 module using a Wi-Fi connection rather than a serial port. Such functionality becomes extremely useful in case of limited or no physical access to the module.

OTA may be done using:

  • Arduino IDE
  • Web Browser
  • HTTP Server

First of all, check the tutorial “ESP32: flash compiled firmware (.bin)“.

Custom web interface

Custom Arduino OTA page
Custom Arduino OTA page

I already explain how to manage WebPage and how to do a REST call, now we are going to manage the existing endpoint of ArduinoOTA with a custom WebPage.

First of all, we are going to analyze the original web page.

Original ArduinoOTA page description

The HTTPUpdateServer wraps WebServer and add an endpoint that, in GET response with a simple page

static const char serverIndex[] PROGMEM =
  R"(<!DOCTYPE html>
     <html lang='en'>
     <head>
         <meta charset='utf-8'>
         <meta name='viewport' content='width=device-width,initial-scale=1'/>
     </head>
     <body>
     <form method='POST' action='' enctype='multipart/form-data'>
         Firmware:<br>
         <input type='file' accept='.bin,.bin.gz' name='firmware'>
         <input type='submit' value='Update Firmware'>
     </form>
     <form method='POST' action='' enctype='multipart/form-data'>
         FileSystem:<br>
         <input type='file' accept='.bin,.bin.gz' name='filesystem'>
         <input type='submit' value='Update FileSystem'>
     </form>
     </body>
     </html>)";
static const char successResponse[] PROGMEM = 
  "<META http-equiv=\"refresh\" content=\"15;URL=/\">Update Success! Rebooting...";

Here the handle

    // handler for the /update form page
    _server->on(path.c_str(), HTTP_GET, [&](){
      if(_username != emptyString && _password != emptyString && !_server->authenticate(_username.c_str(), _password.c_str()))
        return _server->requestAuthentication();
      _server->send_P(200, PSTR("text/html"), serverIndex);
    });

Original ArduinoOTA endpoint description

In POST, check the name of input (firmware or filesystem) to understand if the file is for FLASH or FileSystem, then upload the file and Update the service to do the work. Here is the POST code.

    // handler for the /update form POST (once file upload finishes)
    _server->on(path.c_str(), HTTP_POST, [&](){
      if(!_authenticated)
        return _server->requestAuthentication();
      if (Update.hasError()) {
        _server->send(200, F("text/html"), String(F("Update error: ")) + _updaterError);
      } else {
        _server->client().setNoDelay(true);
        _server->send_P(200, PSTR("text/html"), successResponse);
        delay(100);
        _server->client().stop();
        ESP.restart();
      }
    },[&](){
      // handler for the file upload, gets the sketch bytes, and writes
      // them through the Update object
      HTTPUpload& upload = _server->upload();

      if(upload.status == UPLOAD_FILE_START){
        _updaterError.clear();
        if (_serial_output)
          Serial.setDebugOutput(true);

        _authenticated = (_username == emptyString || _password == emptyString || _server->authenticate(_username.c_str(), _password.c_str()));
        if(!_authenticated){
          if (_serial_output)
            Serial.printf("Unauthenticated Update\n");
          return;
        }

        WiFiUDP::stopAll();
        if (_serial_output)
          Serial.printf("Update: %s\n", upload.filename.c_str());
        if (upload.name == "filesystem") {
          size_t fsSize = ((size_t) &_FS_end - (size_t) &_FS_start);
          close_all_fs();
          if (!Update.begin(fsSize, U_FS)){//start with max available size
            if (_serial_output) Update.printError(Serial);
          }
        } else {
          uint32_t maxSketchSpace = (ESP.getFreeSketchSpace() - 0x1000) & 0xFFFFF000;
          if (!Update.begin(maxSketchSpace, U_FLASH)){//start with max available size
            _setUpdaterError();
          }
        }
      } else if(_authenticated && upload.status == UPLOAD_FILE_WRITE && !_updaterError.length()){
        if (_serial_output) Serial.printf(".");
        if(Update.write(upload.buf, upload.currentSize) != upload.currentSize){
          _setUpdaterError();
        }
      } else if(_authenticated && upload.status == UPLOAD_FILE_END && !_updaterError.length()){
        if(Update.end(true)){ //true to set the size to the current progress
          if (_serial_output) Serial.printf("Update Success: %u\nRebooting...\n", upload.totalSize);
        } else {
          _setUpdaterError();
        }
        if (_serial_output) Serial.setDebugOutput(false);
      } else if(_authenticated && upload.status == UPLOAD_FILE_ABORTED){
        Update.end();
        if (_serial_output) Serial.println("Update was aborted");
      }
      delay(0);
    });

The page does a submit without action

     <form method='POST' action='' enctype='multipart/form-data'>

so if the page URL is http://esp32-webupdate.local/update the post point to the same URL, the difference is that the page came in GET the submit of the data is in POST.

So _server->on(path.c_str(), HTTP_GET, [&](){ serve the web page and _server->on(path.c_str(), HTTP_POST, [&](){ manage the submit of the form data.

The POST manages firmware and filesystem binary,

        if (upload.name == "filesystem") {
          size_t fsSize = ((size_t) &_FS_end - (size_t) &_FS_start);
          close_all_fs();
          if (!Update.begin(fsSize, U_FS)){//start with max available size
            if (_serial_output) Update.printError(Serial);
          }
        } else {
          uint32_t maxSketchSpace = (ESP.getFreeSketchSpace() - 0x1000) & 0xFFFFF000;
          if (!Update.begin(maxSketchSpace, U_FLASH)){//start with max available size
            _setUpdaterError();
          }
        }

and use the name of the input file

<input type='file' accept='.bin,.bin.gz' name='firmware'>

name equal firmware to save sketch compiled binary

<input type='file' accept='.bin,.bin.gz' name='filesystem'>

name equal filesytem to save the filesystem binary. The upload is managed via standard multipart/form-data.

Web pages

For this test, I create 2 web pages all in vanilla js and CSS.

You can find a simple project with these pages here on GitHub.

Download the project with the relative link or via GitHub client.

In the main directory of the project, launch these commands

npm i
npm run dist 

Now you can find in minified directory these files:

  • index_black_white.html
  • index_black_white.html.gz
  • index_black_white_ita.html
  • index_black_white_ita.html.gz
  • index_color.html
  • index_color.html.gz
  • index_color_ita.html
  • index_color_ita.html.gz

The result of these 2 simple pages is this.

Color version page

ArduinoOTA custom color web page
ArduinoOTA custom color web page

Here the page

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Upload OTA data</title>
    <style type="text/css">
        * {
            box-sizing: border-box;
        }

        body {
        @import url("https://fonts.googleapis.com/css?family=Roboto:400,400i,700");
            font-family: Roboto, sans-serif;

            text-align: center;
            min-width: 360px;
        }

        .button-send {
            width: 100%;
            display: inline-block;
            padding: 5px 55px;
            font-size: 16px;
            cursor: pointer;
            text-align: center;
            text-decoration: none;
            outline: none;
            color: #fff;
            background-color: #337ab7;
            border: none;
            border-radius: 15px;
            box-shadow: 2px 2px #999;
        }

        .button-send:hover {
            background-color: #2e6da4
        }

        .button-send:active {
            background-color: #337ab7;
            box-shadow: 1px 1px #666;
            transform: translateY(4px);
        }
        .button-send:disabled {
            background-color: #9c9c9c;
            box-shadow: 1px 1px #666;
            transform: translateY(4px);
        }

        .container {
            padding-top: 80px;
            display: flex;
            justify-content: center;
        }

        .form-container {
            min-width: 380px;
        }

        .progress {
            width: 100%;
            height: 8px;
            background: #e1e4e8;
            border-radius: 4px;
            overflow: hidden;
        }

        .progress .progress-bar {
            display: block;
            height: 100%;
            background: linear-gradient(90deg, #ffd33d, #ea4aaa 17%, #b34bff 34%, #01feff 51%, #ffd33d 68%, #ea4aaa 85%, #b34bff);
            background-size: 300% 100%;
            -webkit-animation: progress-animation 2s linear infinite;
            animation: progress-animation 2s linear infinite;
        }

        .input-file {
            width: 100%;
            border: 0px transparent;
            padding: 4px 4px 4px 4px;
            margin-top: 20px;
            margin-bottom: 20px;
            border-radius: 7px;
            background-color: rgba(51, 122, 183, 0.3);
        }

        .additional-info {
            height: 67px;
        }

        @-webkit-keyframes progress-animation {
            0% {
                background-position: 100%;
            }
            100% {
                background-position: 0;
            }
        }

        @keyframes progress-animation {
            0% {
                background-position: 100%;
            }
            100% {
                background-position: 0;
            }
        }

        .tile-container {
            display: flex;
            justify-content: center;
        }

        .title-image {
            background-image: url();
            background-repeat: no-repeat;
            background-position: center;
            background-size: 40px;
            width: 40px;
            margin-right: 20px;
        }

        .footer-desc {
            position: fixed;
            display: flex;
            bottom: 20px;
            right: 20px;
            align-items: center;
        }

        .footer-link {
            font-size: x-small;
            color: black;
            text-decoration: none;
            padding-left: 10px;
        }

        .button-o {
            cursor: pointer;
            height: 25px;
            font-size: 15px;
            background: none;
            outline: none;
            border: 1px solid rgba(0, 0, 0, 0.35);
            width: 50%;
            background: rgba(51, 122, 183, 0.3);
        }

        .button-o.selected {
            background-color: #337ab7;
            color: white;
        }

        .button-o.left {
            border-radius: 14px 0px 0px 14px;
            margin-right: -3px;
        }

        .button-o.right {
            border-radius: 0px 14px 14px 0px;
            margin-left: -3px;
        }

        #overlay {
            position: fixed; /* Sit on top of the page content */
            display: none; /* Hidden by default */
            width: 100%; /* Full width (cover the whole page) */
            height: 100%; /* Full height (cover the whole page) */
            top: 0;
            left: 0;
            right: 0;
            bottom: 0;
            background-color: rgba(0, 0, 0, 0.06); /* Black background with opacity */
            z-index: 2; /* Specify a stack order in case you're using a different order for other elements */
            cursor: pointer; /* Add a pointer on hover */
        }

    </style>
</head>
<body>
<div id="overlay"></div>
<div class="footer-desc"><img
        src=""/><a
        class="footer-link" href="mischianti.org">www.mischianti.org</a></div>

<div class="tile-container">
    <div class="title-image"></div>
    <h1>OTA Update</h1>
</div>

<div class="container">

    <form enctype="multipart/form-data" class="form-container" id="upload_form" method="post">
        <div id="switch-container">
            <button class="button-o left selected" id="firmware-button" type="button">Firmware</button>
            <button class="button-o right" id="filesystem-button" type="button">FileSystem</button>
        </div>

        <input accept='.bin,.bin.gz' class="input-file" id="file1" name='firmware' type="file"><br>
        <div class="progress">
            <span class="progress-bar" id="progressBar" style="width: 0%"></span>
        </div>
        <div class="additional-info">
            <h3 id="status">Firmware upload</h3>
            <p id="loaded_n_total"></p>
        </div>
        <hr/>
        <button id="button-send" class="button-send" type="submit" disabled>Upload</button>
    </form>
    <script type="application/javascript">
        function stringToBoolean(string){
            switch(string.toLowerCase().trim()){
                case "true": case "yes": case "1": return true;
                case "false": case "no": case "0": case null: return false;
                default: return Boolean(string);
            }
        }

        const urlParams = new URLSearchParams(window.location.search);
        const onlyFirmware = urlParams.get('onlyFirmware');

        if (onlyFirmware && stringToBoolean(onlyFirmware)===true){
            _('switch-container').style.display = 'none';
        }

        function disableAll() {
            document.getElementById("overlay").style.display = "block";
        }

        function enableAll() {
            document.getElementById("overlay").style.display = "none";
        }

        function _(el) {
            return document.getElementById(el);
        }

        function uploadFile() {
            var file = _("file1").files[0];
            // alert(file.name+" | "+file.size+" | "+file.type);
            var formdata = new FormData();
            formdata.append(_("file1").name, file, file.name);
            var ajax = new XMLHttpRequest();
            ajax.upload.addEventListener("progress", progressHandler, false);
            ajax.addEventListener("load", completeHandler, false);
            ajax.addEventListener("loadstart", startHandler, false);
            ajax.addEventListener("error", errorHandler, false);
            ajax.addEventListener("abort", abortHandler, false);
            ajax.open("POST", "/update"); // http://www.developphp.com/video/JavaScript/File-Upload-Progress-Bar-Meter-Tutorial-Ajax-PHP
            //use file_upload_parser.php from above url
            ajax.setRequestHeader('Access-Control-Allow-Headers', '*');
            ajax.setRequestHeader('Access-Control-Allow-Origin', '*');

            ajax.send(formdata);
        }

        function progressHandler(event) {
            _("loaded_n_total").innerHTML = "Uploaded " + event.loaded + " bytes of " + event.total;
            var percent = Math.round((event.loaded / event.total) * 100);
            _("progressBar").style = 'width: ' + percent + '%';
            _("status").innerHTML = percent + "% uploaded... please wait";
        }

        function completeHandler(event) {
            enableAll();
            if (event.target.responseText.indexOf('error')>=0){
                _("status").innerHTML = event.target.responseText;
            }else {
                _("status").innerHTML = 'Upload Success!'; //event.target.responseText;
            }
            _("progressBar").value = 0; //wil clear progress bar after successful upload
        }

        function startHandler(event) {
            disableAll();
        }

        function errorHandler(event) {
            enableAll();
            _("status").innerHTML = "Upload Failed";
        }

        function abortHandler(event) {
            enableAll();
            _("status").innerHTML = "Upload Aborted";
        }


        _('upload_form').addEventListener('submit', (e) => {
            e.preventDefault();
            uploadFile();
        });

        _('firmware-button').addEventListener('click',
            function (e) {
                e.target.classList.add('selected');
                _('filesystem-button').classList.remove('selected');
                _("file1").name = 'firmware';
            }
        )
        _('filesystem-button').addEventListener('click',
            function (e) {
                e.target.classList.add('selected');
                _('firmware-button').classList.remove('selected');
                _("file1").name = 'filesystem';
            }
        )
        _('file1').addEventListener('change', function(e){
            var file = _("file1").files[0];
            if (file && file.name){
                _('button-send').disabled = false;
            }else{
                _('button-send').disabled = true;
            }
            _('status').innerHTML = "Firmware Upload!";
            _("loaded_n_total").innerHTML = "";
        });
    </script>
</div>
</body>
</html>

Black white page

ArduinoOTA custom black white web page
ArduinoOTA custom black white web page

And here the code

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Upload OTA data</title>
    <style type="text/css">
        * {
            box-sizing: border-box;
        }

        body {
        @import url("https://fonts.googleapis.com/css?family=Roboto:400,400i,700");
            font-family: Roboto, sans-serif;
            background: #555;

            text-align: center;
            min-width: 360px;
        }

        .button-send {
            width: 100%;
            border: 0;
            background: #FFF;
            line-height: 23px;
            font-weight: bold;
            color: #555;
            border-radius: 4px;
            box-shadow: inset 0 -2px 3px rgba(0,0,0,.4), 0 2px 5px rgba(0,0,0,0.5);
        }

        .button-send:hover {
            background-color: #dedede;
        }

        .button-send:active {
            background-color: #dedede;
            box-shadow: 1px 1px #666;
            transform: translateY(4px);
        }
        .button-send:disabled {
            background-color: #9c9c9c;
        }

        .container {
            padding-top: 80px;
            display: flex;
            justify-content: center;
        }

        .form-container {
            min-width: 380px;
        }

        progress {
            display: block; /* default: inline-block */
            width: 100%;
            margin: 2em auto;
            padding: 4px;
            border: 0 none;
            background: #444;
            border-radius: 14px;
            box-shadow: inset 0px 1px 1px rgba(0,0,0,0.5), 0px 1px 0px rgba(255,255,255,0.2);
        }
        progress::-moz-progress-bar {
            border-radius: 12px;
            background: #FFF;
            box-shadow: inset 0 -2px 4px rgba(0,0,0,0.4), 0 2px 5px 0px rgba(0,0,0,0.3);

        }
        /* webkit */
        @media screen and (-webkit-min-device-pixel-ratio:0) {
            progress {
                height: 25px;
            }
        }
        progress::-webkit-progress-bar {
            background: transparent;
        }
        progress::-webkit-progress-value {
            border-radius: 12px;
            background: #FFF;
            box-shadow: inset 0 -2px 4px rgba(0,0,0,0.4), 0 2px 5px 0px rgba(0,0,0,0.3);
        }
        /* environnement styles */

        h1 {
            color: #eee;
            font: 50px Helvetica, Arial, sans-serif;
            text-shadow: 0px 1px black;
            text-align: center;
            -webkit-font-smoothing: antialiased;
        }

        .input-file {
            width: 100%;
            border: 0px transparent;
            padding: 4px 4px 4px 4px;
            margin-top: 20px;
            margin-bottom: 0px;
            border-radius: 7px;
            background-color: rgb(255 255 255);
        }

        .additional-info {
            color: white;
            height: 67px;
        }

        @-webkit-keyframes progress-animation {
            0% {
                background-position: 100%;
            }
            100% {
                background-position: 0;
            }
        }

        @keyframes progress-animation {
            0% {
                background-position: 100%;
            }
            100% {
                background-position: 0;
            }
        }

        .tile-container {
            display: flex;
            justify-content: center;
        }

        .title-image {
            background-image: url();
            background-repeat: no-repeat;
            background-position: center;
            background-size: 40px;
            width: 40px;
            margin-right: 20px;
        }

        .footer-desc {
            position: fixed;
            display: flex;
            bottom: 20px;
            right: 20px;
            align-items: center;
        }

        .footer-link {
            font-size: x-small;
            color: black;
            text-decoration: none;
            padding-left: 10px;
        }

        .button-o {
            cursor: pointer;
            height: 25px;
            font-size: 15px;
            background: none;
            outline: none;
            border: 1px solid rgba(0, 0, 0, 0.35);
            width: 50%;
            color: white;
            background: black;
        }

        .button-o.selected {
            background-color: white;
            color: black;
        }

        .button-o.left {
            border-radius: 14px 0px 0px 14px;
            margin-right: -3px;
        }

        .button-o.right {
            border-radius: 0px 14px 14px 0px;
            margin-left: -3px;
        }

        #overlay {
            position: fixed; /* Sit on top of the page content */
            display: none; /* Hidden by default */
            width: 100%; /* Full width (cover the whole page) */
            height: 100%; /* Full height (cover the whole page) */
            top: 0;
            left: 0;
            right: 0;
            bottom: 0;
            background-color: rgba(0, 0, 0, 0.06); /* Black background with opacity */
            z-index: 2; /* Specify a stack order in case you're using a different order for other elements */
            cursor: pointer; /* Add a pointer on hover */
        }

    </style>
</head>
<body>
<div id="overlay"></div>
<div class="footer-desc"><img
        src=""/><a
        class="footer-link" href="mischianti.org">www.mischianti.org</a></div>

<div class="tile-container">
    <div class="title-image"></div>
    <h1>OTA Update</h1>
</div>

<div class="container">

    <form enctype="multipart/form-data" class="form-container" id="upload_form" method="post">
        <div id="switch-container">
            <button class="button-o left selected" id="firmware-button" type="button">Firmware</button>
            <button class="button-o right" id="filesystem-button" type="button">FileSystem</button>
        </div>

        <input accept='.bin,.bin.gz' class="input-file" id="file1" name='firmware' type="file"><br>
        <progress id="progressBar" max="100" value="0"></progress>
        <div class="additional-info">
            <h3 id="status">Firmware upload</h3>
            <p id="loaded_n_total"></p>
        </div>
        <hr/>
        <button id="button-send" class="button-send" type="submit" disabled>Upload</button>
    </form>
    <script type="application/javascript">
        function stringToBoolean(string){
            switch(string.toLowerCase().trim()){
                case "true": case "yes": case "1": return true;
                case "false": case "no": case "0": case null: return false;
                default: return Boolean(string);
            }
        }

        const urlParams = new URLSearchParams(window.location.search);
        const onlyFirmware = urlParams.get('onlyFirmware');

        if (onlyFirmware && stringToBoolean(onlyFirmware)===true){
            _('switch-container').style.display = 'none';
        }

        function disableAll() {
            document.getElementById("overlay").style.display = "block";
        }

        function enableAll() {
            document.getElementById("overlay").style.display = "none";
        }

        function _(el) {
            return document.getElementById(el);
        }

        function uploadFile() {
            var file = _("file1").files[0];
            // alert(file.name+" | "+file.size+" | "+file.type);
            var formdata = new FormData();
            formdata.append(_("file1").name, file, file.name);
            var ajax = new XMLHttpRequest();
            ajax.upload.addEventListener("progress", progressHandler, false);
            ajax.addEventListener("load", completeHandler, false);
            ajax.addEventListener("loadstart", startHandler, false);
            ajax.addEventListener("error", errorHandler, false);
            ajax.addEventListener("abort", abortHandler, false);
            ajax.open("POST", "/update"); // http://www.developphp.com/video/JavaScript/File-Upload-Progress-Bar-Meter-Tutorial-Ajax-PHP
            //use file_upload_parser.php from above url
            ajax.setRequestHeader('Access-Control-Allow-Headers', '*');
            ajax.setRequestHeader('Access-Control-Allow-Origin', '*');

            ajax.send(formdata);
        }

        function progressHandler(event) {
            _("loaded_n_total").innerHTML = "Uploaded " + event.loaded + " bytes of " + event.total;
            var percent = Math.round((event.loaded / event.total) * 100);
            _("progressBar").value = percent;
            _("status").innerHTML = percent + "% uploaded... please wait";
        }

        function completeHandler(event) {
            enableAll();
            _("status").innerHTML = 'Upload Success!'; //event.target.responseText;
            _("progressBar").value = 0; //wil clear progress bar after successful upload
        }

        function startHandler(event) {
            disableAll();
        }

        function errorHandler(event) {
            enableAll();
            _("status").innerHTML = "Upload Failed";
        }

        function abortHandler(event) {
            enableAll();
            _("status").innerHTML = "Upload Aborted";
        }


        _('upload_form').addEventListener('submit', (e) => {
            e.preventDefault();
            uploadFile();
        });

        _('firmware-button').addEventListener('click',
            function (e) {
                e.target.classList.add('selected');
                _('filesystem-button').classList.remove('selected');
                _("file1").name = 'firmware';
            }
        )
        _('filesystem-button').addEventListener('click',
            function (e) {
                e.target.classList.add('selected');
                _('firmware-button').classList.remove('selected');
                _("file1").name = 'filesystem';
            }
        )
        _('file1').addEventListener('change', function(e){
            var file = _("file1").files[0];
            if (file && file.name){
                _('button-send').disabled = false;
                _("progressBar").value = 0; //wil clear progress bar after successful upload
            }else{
                _('button-send').disabled = true;
            }
            _('status').innerHTML = "Firmware Upload!";
            _("loaded_n_total").innerHTML = "";
        });
    </script>
</div>
</body>
</html>

Code description

The code is quite simple, I use standard XMLHttpRequest to execute POST and I change the target (firmware or filesystem).

The form remains very simple.

    <form enctype="multipart/form-data" class="form-container" id="upload_form" method="post">
        <div id="switch-container">
            <button class="button-o left selected" id="firmware-button" type="button">Firmware</button>
            <button class="button-o right" id="filesystem-button" type="button">FileSystem</button>
        </div>

        <input accept='.bin,.bin.gz' class="input-file" id="file1" name='firmware' type="file"><br>
        <progress id="progressBar" max="100" value="0"></progress>
        <div class="additional-info">
            <h3 id="status">Firmware upload</h3>
            <p id="loaded_n_total"></p>
        </div>
        <hr/>
        <button id="button-send" class="button-send" type="submit" disabled>Upload</button>
    </form>

You can see that there isn’t the name of the input “filesystem” this because I manage It via JavaScript. The submit is stopped with this listener.

        _('upload_form').addEventListener('submit', (e) => {
            e.preventDefault();
            uploadFile();
        });

Then the uploadFile function is called.

        function uploadFile() {
            var file = _("file1").files[0];
            var formdata = new FormData();
            formdata.append(_("file1").name, file, file.name);
            var ajax = new XMLHttpRequest();
            ajax.upload.addEventListener("progress", progressHandler, false);
            ajax.addEventListener("load", completeHandler, false);
            ajax.addEventListener("loadstart", startHandler, false);
            ajax.addEventListener("error", errorHandler, false);
            ajax.addEventListener("abort", abortHandler, false);
            ajax.open("POST", "/update");
            // Enable CORS for testing
            ajax.setRequestHeader('Access-Control-Allow-Headers', '*');
            ajax.setRequestHeader('Access-Control-Allow-Origin', '*');

            ajax.send(formdata);
        }

I create a new FormData and add only the file, then with the handlers

  • “progress”: I manage the scrollbar;
  • “load”: manage the result;
  • “loadstart”: I put an overlay to prevent clicking;
  • “error” and “abort”: to manage errors.

Manage custom OTA page

We have 2 options to add this page to the sketch fully explained in this article “Web Server with esp8266 and esp32: byte array, gzipped pages and SPIFFS“, add It to the filesystem or add it as bytearray. I use gzipped for all methods, and you can check the simple version in the linked article.

Add page to filesystem

To upload content to the filesystem, you must install the relative plugin, select one of this

So we are going to add a data folder to the sketch, and we will put the gzipped page we want, now, i put

index_color.html.gz
index_black_white.html.gz

for testing purposes. After that operation, you can use the plugin like usual to upload the pages.

Arduino IDE esp32 SPIFFS Sketch Data Upload
Arduino IDE esp32 SPIFFS Sketch Data Upload
esp32 SPIFFS LittleFS FatFS file uploader from Arduino IDE
esp32 SPIFFS LittleFS FatFS file uploader from Arduino IDE

You can check the IDE console output to check what happened.

Chip : esp32
Using partition scheme from Arduino IDE.
Start: 0x290000
Size : 0x170000
mkspiffs : C:\Users\renzo\AppData\Local\Arduino15\packages\esp32\tools\mkspiffs\0.2.3\mkspiffs.exe

espota : C:\Users\renzo\AppData\Local\Arduino15\packages\esp32\hardware\esp32\2.0.0\tools\espota.exe

[SPIFFS] data   : D:\Projects\Arduino\sloeber-workspace-OTA\ArduinoOTAesp32_basic_arduino\data
[SPIFFS] offset : 0
[SPIFFS] start  : 2686976
[SPIFFS] size   : 1472
[SPIFFS] page   : 256
[SPIFFS] block  : 4096
->/version.txt
[SPIFFS] upload : C:\Users\renzo\AppData\Local\Temp\arduino_build_258074/ArduinoOTAesp32_basic_arduino.spiffs.bin
[SPIFFS] IP     : 192.168.1.186
Running: C:\Users\renzo\AppData\Local\Arduino15\packages\esp32\hardware\esp32\2.0.0\tools\espota.exe -i 192.168.1.186 -p 3232 -s -f C:\Users\renzo\AppData\Local\Temp\arduino_build_258074/ArduinoOTAesp32_basic_arduino.spiffs.bin

_>Sending invitation to 192.168.1.186 
_>Uploading................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................
_>09:21:38 [ERROR]: Error response from device
SPIFFS Upload failed!

The IDE console returns an error, but the upload work, I think there is a little bug now when I write this article.

IDE gives an error but it is a bug, the loading works correctly

You can retrieve the file system bin file from this line

[SPIFFS] upload : C:\Users\renzo\AppData\Local\Temp\arduino_build_258074/ArduinoOTAesp32_basic_arduino.spiffs.bin

Sketch with SPIFFS gzipped custom OTA page

Now we are going to change the WebUpdater.ino file.

/*
 *	Custom OTA page from SPIFFS file system
 *	on ESP32
 *
 *	Renzo Mischianti <www.mischianti.org>
 *
 *	www.mischianti.org
 */

#include <WiFi.h>
#include <WiFiClient.h>
#include <WebServer.h>
#include <ESPmDNS.h>
#include <HTTPUpdateServer.h>
#include <SPIFFS.h>

#ifndef STASSID
#define STASSID "<YOUR-SSID>"
#define STAPSK  "<YOUR-PASSWD>"
#endif

const char* host = "esp32-webupdate";
const char* ssid = STASSID;
const char* password = STAPSK;

WebServer httpServer(80);
HTTPUpdateServer httpUpdater;
void handleNotFound() {
  String message = "File Not Found\n\n";
  message += "URI: ";
  message += httpServer.uri();
  message += "\nMethod: ";
  message += (httpServer.method() == HTTP_GET) ? "GET" : "POST";
  message += "\nArguments: ";
  message += httpServer.args();
  message += "\n";

  for (uint8_t i = 0; i < httpServer.args(); i++) {
    message += " " + httpServer.argName(i) + ": " + httpServer.arg(i) + "\n";
  }

  httpServer.send(404, "text/plain", message);
}

bool loadFromFS(String path, String dataType) {
  Serial.print("Requested page -> ");
  Serial.println(path);
  if (SPIFFS.exists(path)){
      File dataFile = SPIFFS.open(path, "r");
      if (!dataFile) {
          handleNotFound();
          return false;
      }

      if (httpServer.streamFile(dataFile, dataType) != dataFile.size()) {
        Serial.println("Sent less data than expected!");
      }else{
          Serial.println("Page served!");
      }

      dataFile.close();
  }else{
      handleNotFound();
      return false;
  }
  return true;
}

void handleRoot() {
//	loadFromFS("/index_color.html.gz", "text/html");
	loadFromFS("/index_black_white.html.gz", "text/html");
}

void setup(void) {

  Serial.begin(115200);
  Serial.println();
  Serial.println("Booting Sketch...");
  WiFi.mode(WIFI_AP_STA);
  WiFi.begin(ssid, password);

  while (WiFi.waitForConnectResult() != WL_CONNECTED) {
    WiFi.begin(ssid, password);
    Serial.println("WiFi failed, retrying.");
  }

  MDNS.begin(host);
  if (MDNS.begin("esp32")) {
    Serial.println("mDNS responder started");
  }

  Serial.print(F("Inizializing FS..."));
  if (SPIFFS.begin()){
    Serial.println(F("done."));
  }else{
    Serial.println(F("fail."));
  }

  // handler for the /update form page
  httpServer.on("/update", HTTP_GET, handleRoot);

  httpUpdater.setup(&httpServer, "/update", "mischianti", "passwd");
  httpServer.begin();

  MDNS.addService("http", "tcp", 80);

  IPAddress ip = WiFi.localIP();
  Serial.printf("HTTPUpdateServer ready! Open http://%s.local/update in your browser (%u.%u.%u.%u)\n", host, ip & 0xFF, (ip>>8) & 0xFF, (ip>>16) & 0xFF, (ip>>24) & 0xFF);
}

void loop(void) {
  httpServer.handleClient();
}

You can see the code explained in the linked article, and with

  httpServer.on("/update", HTTP_GET, handleRoot);

I override the standard page on /update GET, the handle

bool loadFromFS(String path, String dataType) {
  Serial.print("Requested page -> ");
  Serial.println(path);
  if (LittleFS.exists(path)){
      File dataFile = LittleFS.open(path, "r");
      if (!dataFile) {
          handleNotFound();
          return false;
      }

      if (httpServer.streamFile(dataFile, dataType) != dataFile.size()) {
        Serial.println("Sent less data than expected!");
      }else{
          Serial.println("Page served!");
      }

      dataFile.close();
  }else{
      handleNotFound();
      return false;
  }
  return true;
}

void handleRoot() {
//	loadFromFS("/index_color.html.gz", "text/html");
	loadFromFS("/index_black_white.html.gz", "text/html");
}

read the selected page in handleRoot from SPIFFS and return the stream.

Add custom page as byte array

Get ready to use the page

The article linked before is explained in deep the byte array page management, now in the repository of the pages you can grab the page with HTML extension,

https://github.com/xreef/ArduinoOTA_reusable_custom_web_page/minified

and put It in the bytearray converter by selecting the correct file (like index_color.html), and taking the generated gzipped filearray.

ArduinoOTA custom web page generate gzipped bytearray
ArduinoOTA custom web page generate gzipped bytearray

Sketch with byte array gzipped custom OTA page

Then we can go to modify the WebUpdater.ino to response the selected page.

/*
 *	Custom OTA page from bytearray file system
 *	on ESP32
 *
 *	Renzo Mischianti <www.mischianti.org>
 *
 *	www.mischianti.org
 */
#include <WiFi.h>
#include <WiFiClient.h>
#include <WebServer.h>
#include <ESPmDNS.h>
#include <HTTPUpdateServer.h>

#ifndef STASSID
#define STASSID "<YOUR-SSID>"
#define STAPSK  "<YOUR-PASSWD>"
#endif

const char* host = "esp32-webupdate";
const char* ssid = STASSID;
const char* password = STAPSK;

WebServer httpServer(80);
HTTPUpdateServer httpUpdater;

//File: index_color.html.gz, Size: 33264
#define index_color_html_gz_len 33264
const uint8_t index_color_html_gz[] PROGMEM = {
0x1F, 0x8B, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xB4, 0xBB, 0xE7, 0x92, 0xF4, 0xD8,
0x75, 0x2D, 0xF8, 0x2A, 0x3D, 0x64, 0x28, 0x42, 0xBA, 0x60, 0x13, 0xDE, 0x75, 0x4B, 0x1A, 0xC1,
0xBB, 0x84, 0x37, 0x09, 0xE0, 0x1F, 0xBC, 0xF7, 0x48, 0x98, 0x64, 0xF0, 0xDD, 0x07, 0x5F, 0x93,
0xA2, 0xC8, 0x2B, 0xFD, 0xB8, 0x33, 0x11, 0x53, 0x15, 0xC8, 0x02, 0x50, 0x07, 0x67, 0x9F, 0xB3,
0xCD, 0xDA, 0x6B, 0x45, 0xA1, 0xFE, 0xF5, 0xFF, 0xCA, 0xA7, 0x6C, 0xBF, 0xE7, 0xA2, 0xDE, 0x87,
0xFE, 0xDF, 0xFF, 0xF5, 0xC7, 0xE7, 0x4F, 0x7D, 0x32, 0x56, 0xFF, 0xF6, 0xBB, 0x62, 0xFC, 0xDD,
0xBF, 0xFF, 0xEB, 0x50, 0xEC, 0xC9, 0x4F, 0x59, 0x9D, 0xAC, 0x5B, 0xB1, 0xFF, 0xDB, 0xEF, 0x7C,
0x4F, 0xFC, 0x99, 0x7A, 0xEE, 0xEE, 0xCD, 0xDE, 0x17, 0xFF, 0xEE, 0xCF, 0xFD, 0x94, 0xE4, 0x3F,

[...]

0x17, 0xF4, 0xDE, 0xE0, 0x8B, 0xFC, 0x18, 0xC7, 0xC4, 0x2D, 0x41, 0xB4, 0xFB, 0x1D, 0x92, 0xF3,
0x32, 0xD1, 0xFD, 0x68, 0x99, 0xF8, 0xB5, 0xFB, 0xF9, 0xE7, 0xEE, 0x3E, 0xC7, 0xFF, 0x02, 0xAD,
0x6F, 0xA3, 0x91, 0x4F, 0x8F, 0xAF, 0x21, 0xC8, 0x4F, 0xC8, 0x3F, 0xFE, 0x45, 0x23, 0xFC, 0x57,
0x3A, 0xFC, 0x1A, 0x2F, 0x3D, 0x29, 0xF3, 0x93, 0x6D, 0xFE, 0x2B, 0x8F, 0x05, 0x48, 0x03, 0xD1,
0xD9, 0x53, 0x34, 0xF3, 0x14, 0x41, 0xFD, 0x5F, 0xDB, 0x3E, 0xEF, 0x40, 0x55, 0xB9, 0x00, 0x00
};


void setup(void) {

  Serial.begin(115200);
  Serial.println();
  Serial.println("Booting Sketch...");
  WiFi.mode(WIFI_AP_STA);
  WiFi.begin(ssid, password);

  while (WiFi.waitForConnectResult() != WL_CONNECTED) {
    WiFi.begin(ssid, password);
    Serial.println("WiFi failed, retrying.");
  }

  MDNS.begin(host);
  if (MDNS.begin("esp32")) {
    Serial.println("mDNS responder started");
  }

  const char*  username = "mischianti";
  const char*  password = "password";

  // handler for the /update form page
  httpServer.on("/update", HTTP_GET, [&](){
		httpServer.sendHeader(F("Content-Encoding"), F("gzip"));
		httpServer.send_P(200, "text/html", (const char*)index_color_html_gz, (int)index_color_html_gz_len);
  });

  httpUpdater.setup(&httpServer, "/update", username, password);
  httpServer.begin();

  MDNS.addService("http", "tcp", 80);

  IPAddress ip = WiFi.localIP();
  Serial.printf("HTTPUpdateServer ready! Open http://%s.local/update in your browser (%u.%u.%u.%u)\n", host, ip & 0xFF, (ip>>8) & 0xFF, (ip>>16) & 0xFF, (ip>>24) & 0xFF);
}

void loop(void) {
  httpServer.handleClient();
}

You must add the complete bytearray of the page, in the code there is only a little part

the core code is very simple,

  httpServer.on("/update", HTTP_GET, [&](){
	  httpServer.sendHeader(F("Content-Encoding"), F("gzip"));
	  httpServer.send_P(200, "text/html", (const char*)index_color_html_gz, (int)index_color_html_gz_len);
  });

the send_P stream all the byte array with Content-Encoding gzip, you can use the non-gzipped byte array, but the size grows.

Thanks

  1. ESP32: pinout, specs and Arduino IDE configuration
  2. ESP32: integrated SPIFFS Filesystem
  3. ESP32: manage multiple Serial and logging
  4. ESP32 practical power saving
    1. ESP32 practical power saving: manage WiFi and CPU
    2. ESP32 practical power saving: modem and light sleep
    3. ESP32 practical power saving: deep sleep and hibernation
    4. ESP32 practical power saving: preserve data, timer and touch wake up
    5. ESP32 practical power saving: external and ULP wake up
    6. ESP32 practical power saving: UART and GPIO wake up
  5. ESP32: integrated LittleFS FileSystem
  6. ESP32: integrated FFat (Fat/exFAT) FileSystem
  7. ESP32-wroom-32
    1. ESP32-wroom-32: flash, pinout, specs and IDE configuration
  8. ESP32-CAM
    1. ESP32-CAM: pinout, specs and Arduino IDE configuration
    2. ESP32-CAM: upgrade CamerWebServer with flash features
  9. ESP32: use ethernet w5500 with plain (HTTP) and SSL (HTTPS)
  10. ESP32: use ethernet enc28j60 with plain (HTTP) and SSL (HTTPS)
  11. How to use SD card with esp32
  12. esp32 and esp8266: FAT filesystem on external SPI flash memory
  1. Firmware and OTA update management
    1. Firmware management
      1. ESP32: flash compiled firmware (.bin)
      2. ESP32: flash compiled firmware and filesystem (.bin) with GUI tools
    2. OTA update with Arduino IDE
      1. ESP32 OTA update with Arduino IDE: filesystem, firmware, and password
    3. OTA update with Web Browser
      1. ESP32 OTA update with Web Browser: firmware, filesystem, and authentication
      2. ESP32 OTA update with Web Browser: upload in HTTPS (SSL/TLS) with self-signed certificate
      3. ESP32 OTA update with Web Browser: custom web interface
    4. Self OTA uptate from HTTP server
      1. ESP32 self OTA update firmware from the server
      2. ESP32 self OTA update firmware from the server with version check
      3. ESP32 self-OTA update in HTTPS (SSL/TLS) with trusted self-signed certificate
    5. Non-standard Firmware update
      1. ESP32 firmware and filesystem update from SD card
      2. ESP32 firmware and filesystem update with FTP client
  1. Integrating LAN8720 with ESP32 for Ethernet Connectivity with plain (HTTP) and SSL (HTTPS)
  2. Connecting the EByte E70 to ESP32 c3/s3 devices and a simple sketch example
  3. ESP32-C3: pinout, specs and Arduino IDE configuration

Spread the love

Leave a Reply

Your email address will not be published. Required fields are marked *