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.
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:
- Arduino IDE
- Web Browser
- Server HTTP
Prima di tutto, controlla il tutorial “ESP32: flash del firmware binario compilato (.bin)“.
Interfaccia web personalizzata
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
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
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:
- ESP32: fileSystem integrato SPIFFS
- ESP32: filesystem integrato LittleFS
- ESP32: filesystem integrato FFat (FAT/exFAT)
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.
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.
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
- ESP32: piedinatura, specifiche e configurazione dell’Arduino IDE
- ESP32: fileSystem integrato SPIFFS
- ESP32: gestire più seriali e logging per il debug
- ESP32 risparmio energetico pratico
- ESP32 risparmio energetico pratico: gestire WiFi e CPU
- ESP32 risparmio energetico pratico: modem e light sleep
- ESP32 risparmio energetico pratico: deep sleep e ibernazione
- ESP32 risparmio energetico pratico: preservare dati al riavvio, sveglia a tempo e tramite tocco
- ESP32 risparmio energetico pratico: sveglia esterna e da ULP
- ESP32 risparmio energetico pratico: sveglia da UART e GPIO
- ESP32: filesystem integrato LittleFS
- ESP32: filesystem integrato FFat (Fat/exFAT)
- ESP32-wroom-32
- ESP32-CAM
- ESP32: ethernet w5500 con chiamate standard (HTTP) e SSL (HTTPS)
- ESP32: ethernet enc28j60 con chiamate standard (HTTP) e SSL (HTTPS)
- Come usare la scheda SD con l’esp32
- esp32 e esp8266: file system FAT su memoria SPI flash esterna
- Gestione aggiornamenti firmware e OTA
- Gestione del firmware
- Aggiornamento OTA con Arduino IDE
- Aggiornamento OTA con browser web
- Aggiornamenti automatici OTA da un server HTTP
- Aggiornamento del firmware non standard
- Integrare LAN8720 con ESP32 per la connettività Ethernet con plain (HTTP) e SSL (HTTPS)
- Collegare l’EByte E70 (CC1310) ai dispositivi ESP32 c3/s3 ed un semplice sketch di esempio
- ESP32-C3: piedinatura, specifiche e configurazione dell’IDE Arduino
- Integrazione del modulo W5500 su ESP32 con Core 3: supporto nativo ai protocolli Ethernet con SSL e altre funzionalità
- Integrazione del modulo LAN8720 su ESP32 con Core 3: supporto nativo del protocollo Ethernet con SSL e altre funzionalità.