Aggiornamenti OTA su ESP32 tramite browser web: interfaccia web personalizzata – 3

Spread the love

ESP32 è un potente microcontrollore che può essere utilizzato per un’ampia gamma di applicazioni Internet of Things (IoT). Una delle caratteristiche principali dell’ESP32 è la sua capacità di ricevere aggiornamenti Over-The-Air (OTA). Gli aggiornamenti OTA ti consentono di aggiornare da remoto il firmware ESP32 senza doverlo collegare fisicamente a un computer o a un cavo USB.

In questo tutorial, esploreremo come implementare gli aggiornamenti OTA su un ESP32 utilizzando un browser Web con un’interfaccia personalizzata. Ciò significa che saremo in grado di caricare gli aggiornamenti del firmware sull’ESP32 tramite una pagina Web ospitata sul dispositivo stesso.

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

Inizieremo impostando le librerie necessarie e configurando l’ESP32 per connettersi a una rete locale. Creeremo quindi un’interfaccia web personalizzata utilizzando HTML e JavaScript, che ci consentirà di cercare e caricare i file del firmware su ESP32.

Here some common devices ESP32 Dev Kit v1 - Selectable - TTGO T-Display 1.14 ESP32 - NodeMCU V3 V2 ESP8266 Lolin32 - NodeMCU ESP-32S - WeMos Lolin32 CP2104 CH340 - 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.

Gli aggiornamenti OTA possono essere effettuati utilizzando:

  • Arduino IDE
  • Web Browser
  • Server HTTP

Prima di tutto, controlla il tutorial “ESP32: flash del firmware binario compilato (.bin)“.

Interfaccia web personalizzata

Custom Arduino OTA page
Custom Arduino OTA page

Ho già spiegato come gestire pagine web e come fare una chiamata REST, ora andremo a gestire l’endpoint esistente di ArduinoOTA con una WebPage personalizzata.

Prima di tutto analizzeremo la pagina web originale.

Descrizione della pagina ArduinoOTA originale

HTTPUpdateServer wrappa l’WebServer e aggiunge un endpoint in GET con una semplice pagina in response

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...";

Qui l’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);
    });

Descrizione degli endpoint ArduinoOTA originali

In POST controlla il nome dell’input (firmware o filesystem) per capire se il file è per FLASH o FileSystem, quindi carica il file e il servizio di aggiornamento fa il suo lavoro, qui il codice della POST.

    // 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);
    });

La pagina esegue un invio senza action

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

quindi se l’url della pagina è http://esp32-webupdate.local/update il post punta allo stesso url, la differenza è che la pagina è arrivata in GET l’invio dei dati è in POST.

Quindi _server->on(path.c_str(), HTTP_GET, [&](){ serve la pagina web e _server->on(path.c_str(), HTTP_POST, [&](){ gestisci l’invio dei dati del modulo.

Il POST gestisce sia firmware che filesystem binari,

        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();
          }
        }

e usa il nome del file di input

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

nome uguale firmware per salvare lo sketch compilato binario

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

nome uguale filesytem per salvare il file system binario. Il caricamento è gestito tramite standard multipart/form-data.

Pagine web

Per questo test creo 2 pagine web tutte in vanilla js e CSS.

Puoi trovare un semplice progetto con queste pagine qui su GitHub .

Scarica il progetto con il relativo link o tramite client GitHub.

Nella directory principale del progetto lancia questi comandi

npm i
npm run dist 

Ora puoi trovare nella directory minimizzata questi file:

  • 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

Il risultato di queste 2 semplici pagine è questo.

Versione a colori

ArduinoOTA custom color web page
ArduinoOTA custom color web page

Qui la pagina

<!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>

Pagina in bianco e nero

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

E qui il codice

<!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>

Descrizione del codice

Il codice è abbastanza semplice, utilizzo lo standard XMLHttpRequest per eseguire il POST e cambio il target (firmware o filesystem).

La form rimane molto semplice.

    <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>

Puoi vedere che non c’è il nome dell’input “filesystem” questo perché lo gestisco tramite JavaScript. L’invio viene interrotto con questo listener.

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

Quindi viene chiamata la funzione uploadFile.

        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);
        }

Creo nuovi FormData e aggiungo solo il file con i gestori

  • “progress”: gestisco la barra di scorrimento;
  • “carica”: gestisci il risultato;
  • “loadstart”: ho messo un overlay per impedire il clic;
  • “error” e “abort” : per gestire gli errori.

Gestire la pagina OTA personalizzata

Abbiamo 2 opzioni per aggiungere questa pagina allo sketch spiegato in questo articolo “Server Web con esp8266 ed esp32: array di byte, pagine gzip e SPIFFS“: puoi aggiungerlo al filesystem o come bytearray. Uso gzipped per tutti i metodi, puoi controllare la versione semplice sull’articolo collegato.

Add page to filesystem

Puoi anche caricare i dati del filesystem con il metodo classico tramite il plugin, per installare il plugin SPIFFS, LittleFS o FFat, fai riferimento al relativo tutorial:

    Quindi aggiungeremo una cartella di dati allo sketch e inseriremo la pagina gzippata che vogliamo, ora, ho inserito

    index_color.html.gz
    index_black_white.html.gz

    a scopo di test. Dopo tale operazione, puoi utilizzare il plug-in come al solito per caricare le pagine.

    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

    Puoi controllare l’output della console dell’IDE per verificare cosa è successo.

    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!
    

    La console dell’IDE restituisce un errore, ma il caricamento funziona, penso che ci sia un piccolo bug ora quando scrivo questo articolo.

    IDE dà un errore ma è un bug, il caricamento funziona correttamente

    È possibile recuperare il file bin del file system da questa riga

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

    Sketch con la pagina OTA personalizzata gzip in SPIFFS

    Ora cambieremo il file WebUpdater.ino.

    /*
     *	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();
    }
    
    

    Puoi vedere il codice spiegato nell’articolo collegato e con

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

    Sovrascrivo la pagina standard sulla GET /update, l’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");
    }
    

    leggere la pagina selezionata in handleRoot dallo SPIFFS e restituire il flusso.

    Aggiungi una pagina personalizzata come array di byte

    Preparati a utilizzare la pagina

    L’articolo linkato prima spiega in modo approfondito la gestione delle pagine in byte array, ora nel repository delle pagine puoi prendere la pagina con estensione HTML,

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

    e inseriscilo nel convertitore bytearray selezionando il file corretto (come index_color.html) e prendendo il filearray generato e compresso con gzip.

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

    Sketch con bytearray pagina OTA personalizzata con gzip

    Quindi possiamo andare a modificare la risposta di WebUpdater.ino alla pagina selezionata.

    /*
     *	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();
    }
    

    Devi aggiungere il bytearray completo della pagina, nel codice c’è solo una piccola parte

    il codice di base è molto semplice,

      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);
      });
    

    il flusso send_P stream tutto l’array di byte con Content-Encoding gzip, è possibile utilizzare l’array di byte non gzippato, ma la dimensione aumenta.

    Grazie

    1. ESP32: piedinatura, specifiche e configurazione dell’Arduino IDE
    2. ESP32: fileSystem integrato SPIFFS
    3. ESP32: gestire più seriali e logging per il debug
    4. ESP32 risparmio energetico pratico
      1. ESP32 risparmio energetico pratico: gestire WiFi e CPU
      2. ESP32 risparmio energetico pratico: modem e light sleep
      3. ESP32 risparmio energetico pratico: deep sleep e ibernazione
      4. ESP32 risparmio energetico pratico: preservare dati al riavvio, sveglia a tempo e tramite tocco
      5. ESP32 risparmio energetico pratico: sveglia esterna e da ULP
      6. ESP32 risparmio energetico pratico: sveglia da UART e GPIO
    5. ESP32: filesystem integrato LittleFS
    6. ESP32: filesystem integrato FFat (Fat/exFAT)
    7. ESP32-wroom-32
      1. ESP32-wroom-32: flash, piedinatura, specifiche e configurazione dell’Arduino IDE
    8. ESP32-CAM
      1. ESP32-CAM: piedinatura, specifiche e configurazione dell’Arduino IDE
      2. ESP32-CAM: upgrade CamerWebServer con gestione della luce flash
    9. ESP32: ethernet w5500 con chiamate standard (HTTP) e SSL (HTTPS)
    10. ESP32: ethernet enc28j60 con chiamate standard (HTTP) e SSL (HTTPS)
    11. Come usare la scheda SD con l’esp32
    12. esp32 e esp8266: file system FAT su memoria SPI flash esterna
    13. Gestione aggiornamenti firmware e OTA
      1. Gestione del firmware
        1. ESP32: flash del firmware binario compilato (.bin)
        2. ESP32: flash del firmware e filesystem (.bin) con strumenti grafici
      2. Aggiornamento OTA con Arduino IDE
        1. Aggiornamenti OTA su ESP32 con Arduino IDE: filesystem, firmware e password
      3. Aggiornamento OTA con browser web
        1. Aggiornamenti OTA su ESP32 tramite browser web: firmware, filesystem e autenticazione
        2. Aggiornamenti OTA su ESP32 tramite browser web: caricamento in HTTPS (SSL/TLS) con certificato autofirmato
        3. Aggiornamenti OTA su ESP32 tramite browser web: interfaccia web personalizzata
      4. Aggiornamenti automatici OTA da un server HTTP
        1. Aggiornamento automatico Firmware OTA dell’ESP32 dal server
        2. Aggiornamento automatico Firmware OTA dell’ESP32 dal server con controllo della versione
        3. Aggiornamento automatico Firmware OTA dell’ESP32 in HTTPS (SSL/TLS) con certificato autofirmato affidabile
      5. Aggiornamento del firmware non standard
        1. Aggiornamento firmware e filesystem ESP32 dalla scheda SD
        2. Aggiornamento firmware e filesystem ESP32 con client FTP
    14. Integrare LAN8720 con ESP32 per la connettività Ethernet con plain (HTTP) e SSL (HTTPS)
    15. i2c esp32: gestione rete a 5v, 3.3v e interfaccia aggiuntiva
    16. […]

    Spread the love

    Lascia un commento

    Il tuo indirizzo email non sarà pubblicato. I campi obbligatori sono contrassegnati *