Site icon Renzo Mischianti

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

Aggiornamento ESP32 OTA con browser web: interfaccia web personalizzata

Aggiornamento ESP32 OTA con browser web: interfaccia web personalizzata

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

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

Gli aggiornamenti OTA possono essere effettuati utilizzando:

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

Interfaccia web personalizzata

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:

Il risultato di queste 2 semplici pagine è questo.

Versione a colori

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

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

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

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. Collegare l’EByte E70 (CC1310) ai dispositivi ESP32 c3/s3 ed un semplice sketch di esempio
  16. ESP32-C3: piedinatura, specifiche e configurazione dell’IDE Arduino
  17. Integrazione del modulo W5500 su ESP32 con Core 3: supporto nativo ai protocolli Ethernet con SSL e altre funzionalità
  18. Integrazione del modulo LAN8720 su ESP32 con Core 3: supporto nativo del protocollo Ethernet con SSL e altre funzionalità.

Spread the love
Exit mobile version