esp8266 OTA update with Web Browser: custom web interface – 3
OTA (Over the Air) update is the process of uploading firmware to an ESP module using a Wi-Fi connection rather than a serial port. Such functionality becomes extremely useful in case of limited or no physical access to the module.
OTA may be done using:
- Arduino IDE
- Web Browser
- HTTP Server
First of all read the tutorial “esp8266: flash firmware binary (.bin) compiled and signed“.
In this article,e we are going to explain OTA updates via Web Browser with a custom web page.
Custom web interface
I already explain how to manage WebPage and how to do a REST calls, now we are going to manage the existing endpoint of ArduinoOTA with a custom WebPage.
First of all we are going to analize the original web page.
Original ArduinoOTA page description
The ESP8266HTTPUpdateServer wrap ESP8266WebServer and add an endpoint that in GET response with a simple page
static const char serverIndex[] PROGMEM =
R"(<!DOCTYPE html>
<html lang='en'>
<head>
<meta charset='utf-8'>
<meta name='viewport' content='width=device-width,initial-scale=1'/>
</head>
<body>
<form method='POST' action='' enctype='multipart/form-data'>
Firmware:<br>
<input type='file' accept='.bin,.bin.gz' name='firmware'>
<input type='submit' value='Update Firmware'>
</form>
<form method='POST' action='' enctype='multipart/form-data'>
FileSystem:<br>
<input type='file' accept='.bin,.bin.gz' name='filesystem'>
<input type='submit' value='Update FileSystem'>
</form>
</body>
</html>)";
static const char successResponse[] PROGMEM =
"<META http-equiv=\"refresh\" content=\"15;URL=/\">Update Success! Rebooting...";
Here the handle
// handler for the /update form page
_server->on(path.c_str(), HTTP_GET, [&](){
if(_username != emptyString && _password != emptyString && !_server->authenticate(_username.c_str(), _password.c_str()))
return _server->requestAuthentication();
_server->send_P(200, PSTR("text/html"), serverIndex);
});
Original ArduinoOTA end points description
In POST check the name of input (firmware or filesystem) to understand if the file is for FLASH or FileSystem, then upload the file and Update service do the work, here the POST code.
// handler for the /update form POST (once file upload finishes)
_server->on(path.c_str(), HTTP_POST, [&](){
if(!_authenticated)
return _server->requestAuthentication();
if (Update.hasError()) {
_server->send(200, F("text/html"), String(F("Update error: ")) + _updaterError);
} else {
_server->client().setNoDelay(true);
_server->send_P(200, PSTR("text/html"), successResponse);
delay(100);
_server->client().stop();
ESP.restart();
}
},[&](){
// handler for the file upload, gets the sketch bytes, and writes
// them through the Update object
HTTPUpload& upload = _server->upload();
if(upload.status == UPLOAD_FILE_START){
_updaterError.clear();
if (_serial_output)
Serial.setDebugOutput(true);
_authenticated = (_username == emptyString || _password == emptyString || _server->authenticate(_username.c_str(), _password.c_str()));
if(!_authenticated){
if (_serial_output)
Serial.printf("Unauthenticated Update\n");
return;
}
WiFiUDP::stopAll();
if (_serial_output)
Serial.printf("Update: %s\n", upload.filename.c_str());
if (upload.name == "filesystem") {
size_t fsSize = ((size_t) &_FS_end - (size_t) &_FS_start);
close_all_fs();
if (!Update.begin(fsSize, U_FS)){//start with max available size
if (_serial_output) Update.printError(Serial);
}
} else {
uint32_t maxSketchSpace = (ESP.getFreeSketchSpace() - 0x1000) & 0xFFFFF000;
if (!Update.begin(maxSketchSpace, U_FLASH)){//start with max available size
_setUpdaterError();
}
}
} else if(_authenticated && upload.status == UPLOAD_FILE_WRITE && !_updaterError.length()){
if (_serial_output) Serial.printf(".");
if(Update.write(upload.buf, upload.currentSize) != upload.currentSize){
_setUpdaterError();
}
} else if(_authenticated && upload.status == UPLOAD_FILE_END && !_updaterError.length()){
if(Update.end(true)){ //true to set the size to the current progress
if (_serial_output) Serial.printf("Update Success: %u\nRebooting...\n", upload.totalSize);
} else {
_setUpdaterError();
}
if (_serial_output) Serial.setDebugOutput(false);
} else if(_authenticated && upload.status == UPLOAD_FILE_ABORTED){
Update.end();
if (_serial_output) Serial.println("Update was aborted");
}
delay(0);
});
The page do a submit without action
<form method='POST' action='' enctype='multipart/form-data'>
so if the page url is http://esp8266-webupdate.local/update
the post point to the same url, the difference is that the page came in GET the submit of the data is in POST.
So _server->on(path.c_str(), HTTP_GET, [&](){
serve the web page and _server->on(path.c_str(), HTTP_POST, [&](){
manage the submit of the form data.
The POST manage firmware and filesystem binary,
if (upload.name == "filesystem") {
size_t fsSize = ((size_t) &_FS_end - (size_t) &_FS_start);
close_all_fs();
if (!Update.begin(fsSize, U_FS)){//start with max available size
if (_serial_output) Update.printError(Serial);
}
} else {
uint32_t maxSketchSpace = (ESP.getFreeSketchSpace() - 0x1000) & 0xFFFFF000;
if (!Update.begin(maxSketchSpace, U_FLASH)){//start with max available size
_setUpdaterError();
}
}
and use the name of input file
<input type='file' accept='.bin,.bin.gz' name='firmware'>
name equal firmware
to save sketch compiled binary
<input type='file' accept='.bin,.bin.gz' name='filesystem'>
name equal filesytem
to save filesystem binary. The upload is managed via standard multipart/form-data
.
Web pages
For this test I create 2 web pages all in vanilla js and CSS.
You can find a simple project with these pages here on GitHub.
Download the project with the relative link or via GitHub client.
In the main directory of the project launch these commands
npm i
npm run dist
Now you can find in minified direcotry these files:
- index_black_white.html
- index_black_white.html.gz
- index_black_white_ita.html
- index_black_white_ita.html.gz
- index_color.html
- index_color.html.gz
- index_color_ita.html
- index_color_ita.html.gz
The result of these 2 simple pages is this.
Color version
Here the page
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Upload OTA data</title>
<style type="text/css">
* {
box-sizing: border-box;
}
body {
@import url("https://fonts.googleapis.com/css?family=Roboto:400,400i,700");
font-family: Roboto, sans-serif;
text-align: center;
min-width: 360px;
}
.button-send {
width: 100%;
display: inline-block;
padding: 5px 55px;
font-size: 16px;
cursor: pointer;
text-align: center;
text-decoration: none;
outline: none;
color: #fff;
background-color: #337ab7;
border: none;
border-radius: 15px;
box-shadow: 2px 2px #999;
}
.button-send:hover {
background-color: #2e6da4
}
.button-send:active {
background-color: #337ab7;
box-shadow: 1px 1px #666;
transform: translateY(4px);
}
.button-send:disabled {
background-color: #9c9c9c;
box-shadow: 1px 1px #666;
transform: translateY(4px);
}
.container {
padding-top: 80px;
display: flex;
justify-content: center;
}
.form-container {
min-width: 380px;
}
.progress {
width: 100%;
height: 8px;
background: #e1e4e8;
border-radius: 4px;
overflow: hidden;
}
.progress .progress-bar {
display: block;
height: 100%;
background: linear-gradient(90deg, #ffd33d, #ea4aaa 17%, #b34bff 34%, #01feff 51%, #ffd33d 68%, #ea4aaa 85%, #b34bff);
background-size: 300% 100%;
-webkit-animation: progress-animation 2s linear infinite;
animation: progress-animation 2s linear infinite;
}
.input-file {
width: 100%;
border: 0px transparent;
padding: 4px 4px 4px 4px;
margin-top: 20px;
margin-bottom: 20px;
border-radius: 7px;
background-color: rgba(51, 122, 183, 0.3);
}
.additional-info {
height: 67px;
}
@-webkit-keyframes progress-animation {
0% {
background-position: 100%;
}
100% {
background-position: 0;
}
}
@keyframes progress-animation {
0% {
background-position: 100%;
}
100% {
background-position: 0;
}
}
.tile-container {
display: flex;
justify-content: center;
}
.title-image {
background-image: url();
background-repeat: no-repeat;
background-position: center;
background-size: 40px;
width: 40px;
margin-right: 20px;
}
.footer-desc {
position: fixed;
display: flex;
bottom: 20px;
right: 20px;
align-items: center;
}
.footer-link {
font-size: x-small;
color: black;
text-decoration: none;
padding-left: 10px;
}
.button-o {
cursor: pointer;
height: 25px;
font-size: 15px;
background: none;
outline: none;
border: 1px solid rgba(0, 0, 0, 0.35);
width: 50%;
background: rgba(51, 122, 183, 0.3);
}
.button-o.selected {
background-color: #337ab7;
color: white;
}
.button-o.left {
border-radius: 14px 0px 0px 14px;
margin-right: -3px;
}
.button-o.right {
border-radius: 0px 14px 14px 0px;
margin-left: -3px;
}
#overlay {
position: fixed; /* Sit on top of the page content */
display: none; /* Hidden by default */
width: 100%; /* Full width (cover the whole page) */
height: 100%; /* Full height (cover the whole page) */
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: rgba(0, 0, 0, 0.06); /* Black background with opacity */
z-index: 2; /* Specify a stack order in case you're using a different order for other elements */
cursor: pointer; /* Add a pointer on hover */
}
</style>
</head>
<body>
<div id="overlay"></div>
<div class="footer-desc"><img
src=""/><a
class="footer-link" href="mischianti.org">www.mischianti.org</a></div>
<div class="tile-container">
<div class="title-image"></div>
<h1>OTA Update</h1>
</div>
<div class="container">
<form enctype="multipart/form-data" class="form-container" id="upload_form" method="post">
<div id="switch-container">
<button class="button-o left selected" id="firmware-button" type="button">Firmware</button>
<button class="button-o right" id="filesystem-button" type="button">FileSystem</button>
</div>
<input accept='.bin,.bin.gz' class="input-file" id="file1" name='firmware' type="file"><br>
<div class="progress">
<span class="progress-bar" id="progressBar" style="width: 0%"></span>
</div>
<div class="additional-info">
<h3 id="status">Firmware upload</h3>
<p id="loaded_n_total"></p>
</div>
<hr/>
<button id="button-send" class="button-send" type="submit" disabled>Upload</button>
</form>
<script type="application/javascript">
function stringToBoolean(string){
switch(string.toLowerCase().trim()){
case "true": case "yes": case "1": return true;
case "false": case "no": case "0": case null: return false;
default: return Boolean(string);
}
}
const urlParams = new URLSearchParams(window.location.search);
const onlyFirmware = urlParams.get('onlyFirmware');
if (onlyFirmware && stringToBoolean(onlyFirmware)===true){
_('switch-container').style.display = 'none';
}
function disableAll() {
document.getElementById("overlay").style.display = "block";
}
function enableAll() {
document.getElementById("overlay").style.display = "none";
}
function _(el) {
return document.getElementById(el);
}
function uploadFile() {
var file = _("file1").files[0];
// alert(file.name+" | "+file.size+" | "+file.type);
var formdata = new FormData();
formdata.append(_("file1").name, file, file.name);
var ajax = new XMLHttpRequest();
ajax.upload.addEventListener("progress", progressHandler, false);
ajax.addEventListener("load", completeHandler, false);
ajax.addEventListener("loadstart", startHandler, false);
ajax.addEventListener("error", errorHandler, false);
ajax.addEventListener("abort", abortHandler, false);
ajax.open("POST", "/update"); // http://www.developphp.com/video/JavaScript/File-Upload-Progress-Bar-Meter-Tutorial-Ajax-PHP
//use file_upload_parser.php from above url
ajax.setRequestHeader('Access-Control-Allow-Headers', '*');
ajax.setRequestHeader('Access-Control-Allow-Origin', '*');
ajax.send(formdata);
}
function progressHandler(event) {
_("loaded_n_total").innerHTML = "Uploaded " + event.loaded + " bytes of " + event.total;
var percent = Math.round((event.loaded / event.total) * 100);
_("progressBar").style = 'width: ' + percent + '%';
_("status").innerHTML = percent + "% uploaded... please wait";
}
function completeHandler(event) {
enableAll();
if (event.target.responseText.indexOf('error')>=0){
_("status").innerHTML = event.target.responseText;
}else {
_("status").innerHTML = 'Upload Success!'; //event.target.responseText;
}
_("progressBar").value = 0; //wil clear progress bar after successful upload
}
function startHandler(event) {
disableAll();
}
function errorHandler(event) {
enableAll();
_("status").innerHTML = "Upload Failed";
}
function abortHandler(event) {
enableAll();
_("status").innerHTML = "Upload Aborted";
}
_('upload_form').addEventListener('submit', (e) => {
e.preventDefault();
uploadFile();
});
_('firmware-button').addEventListener('click',
function (e) {
e.target.classList.add('selected');
_('filesystem-button').classList.remove('selected');
_("file1").name = 'firmware';
}
)
_('filesystem-button').addEventListener('click',
function (e) {
e.target.classList.add('selected');
_('firmware-button').classList.remove('selected');
_("file1").name = 'filesystem';
}
)
_('file1').addEventListener('change', function(e){
var file = _("file1").files[0];
if (file && file.name){
_('button-send').disabled = false;
}else{
_('button-send').disabled = true;
}
_('status').innerHTML = "Firmware Upload!";
_("loaded_n_total").innerHTML = "";
});
</script>
</div>
</body>
</html>
Black white
And here the code
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Upload OTA data</title>
<style type="text/css">
* {
box-sizing: border-box;
}
body {
@import url("https://fonts.googleapis.com/css?family=Roboto:400,400i,700");
font-family: Roboto, sans-serif;
background: #555;
text-align: center;
min-width: 360px;
}
.button-send {
width: 100%;
border: 0;
background: #FFF;
line-height: 23px;
font-weight: bold;
color: #555;
border-radius: 4px;
box-shadow: inset 0 -2px 3px rgba(0,0,0,.4), 0 2px 5px rgba(0,0,0,0.5);
}
.button-send:hover {
background-color: #dedede;
}
.button-send:active {
background-color: #dedede;
box-shadow: 1px 1px #666;
transform: translateY(4px);
}
.button-send:disabled {
background-color: #9c9c9c;
}
.container {
padding-top: 80px;
display: flex;
justify-content: center;
}
.form-container {
min-width: 380px;
}
progress {
display: block; /* default: inline-block */
width: 100%;
margin: 2em auto;
padding: 4px;
border: 0 none;
background: #444;
border-radius: 14px;
box-shadow: inset 0px 1px 1px rgba(0,0,0,0.5), 0px 1px 0px rgba(255,255,255,0.2);
}
progress::-moz-progress-bar {
border-radius: 12px;
background: #FFF;
box-shadow: inset 0 -2px 4px rgba(0,0,0,0.4), 0 2px 5px 0px rgba(0,0,0,0.3);
}
/* webkit */
@media screen and (-webkit-min-device-pixel-ratio:0) {
progress {
height: 25px;
}
}
progress::-webkit-progress-bar {
background: transparent;
}
progress::-webkit-progress-value {
border-radius: 12px;
background: #FFF;
box-shadow: inset 0 -2px 4px rgba(0,0,0,0.4), 0 2px 5px 0px rgba(0,0,0,0.3);
}
/* environnement styles */
h1 {
color: #eee;
font: 50px Helvetica, Arial, sans-serif;
text-shadow: 0px 1px black;
text-align: center;
-webkit-font-smoothing: antialiased;
}
.input-file {
width: 100%;
border: 0px transparent;
padding: 4px 4px 4px 4px;
margin-top: 20px;
margin-bottom: 0px;
border-radius: 7px;
background-color: rgb(255 255 255);
}
.additional-info {
color: white;
height: 67px;
}
@-webkit-keyframes progress-animation {
0% {
background-position: 100%;
}
100% {
background-position: 0;
}
}
@keyframes progress-animation {
0% {
background-position: 100%;
}
100% {
background-position: 0;
}
}
.tile-container {
display: flex;
justify-content: center;
}
.title-image {
background-image: url();
background-repeat: no-repeat;
background-position: center;
background-size: 40px;
width: 40px;
margin-right: 20px;
}
.footer-desc {
position: fixed;
display: flex;
bottom: 20px;
right: 20px;
align-items: center;
}
.footer-link {
font-size: x-small;
color: black;
text-decoration: none;
padding-left: 10px;
}
.button-o {
cursor: pointer;
height: 25px;
font-size: 15px;
background: none;
outline: none;
border: 1px solid rgba(0, 0, 0, 0.35);
width: 50%;
color: white;
background: black;
}
.button-o.selected {
background-color: white;
color: black;
}
.button-o.left {
border-radius: 14px 0px 0px 14px;
margin-right: -3px;
}
.button-o.right {
border-radius: 0px 14px 14px 0px;
margin-left: -3px;
}
#overlay {
position: fixed; /* Sit on top of the page content */
display: none; /* Hidden by default */
width: 100%; /* Full width (cover the whole page) */
height: 100%; /* Full height (cover the whole page) */
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: rgba(0, 0, 0, 0.06); /* Black background with opacity */
z-index: 2; /* Specify a stack order in case you're using a different order for other elements */
cursor: pointer; /* Add a pointer on hover */
}
</style>
</head>
<body>
<div id="overlay"></div>
<div class="footer-desc"><img
src=""/><a
class="footer-link" href="mischianti.org">www.mischianti.org</a></div>
<div class="tile-container">
<div class="title-image"></div>
<h1>OTA Update</h1>
</div>
<div class="container">
<form enctype="multipart/form-data" class="form-container" id="upload_form" method="post">
<div id="switch-container">
<button class="button-o left selected" id="firmware-button" type="button">Firmware</button>
<button class="button-o right" id="filesystem-button" type="button">FileSystem</button>
</div>
<input accept='.bin,.bin.gz' class="input-file" id="file1" name='firmware' type="file"><br>
<progress id="progressBar" max="100" value="0"></progress>
<div class="additional-info">
<h3 id="status">Firmware upload</h3>
<p id="loaded_n_total"></p>
</div>
<hr/>
<button id="button-send" class="button-send" type="submit" disabled>Upload</button>
</form>
<script type="application/javascript">
function stringToBoolean(string){
switch(string.toLowerCase().trim()){
case "true": case "yes": case "1": return true;
case "false": case "no": case "0": case null: return false;
default: return Boolean(string);
}
}
const urlParams = new URLSearchParams(window.location.search);
const onlyFirmware = urlParams.get('onlyFirmware');
if (onlyFirmware && stringToBoolean(onlyFirmware)===true){
_('switch-container').style.display = 'none';
}
function disableAll() {
document.getElementById("overlay").style.display = "block";
}
function enableAll() {
document.getElementById("overlay").style.display = "none";
}
function _(el) {
return document.getElementById(el);
}
function uploadFile() {
var file = _("file1").files[0];
// alert(file.name+" | "+file.size+" | "+file.type);
var formdata = new FormData();
formdata.append(_("file1").name, file, file.name);
var ajax = new XMLHttpRequest();
ajax.upload.addEventListener("progress", progressHandler, false);
ajax.addEventListener("load", completeHandler, false);
ajax.addEventListener("loadstart", startHandler, false);
ajax.addEventListener("error", errorHandler, false);
ajax.addEventListener("abort", abortHandler, false);
ajax.open("POST", "/update"); // http://www.developphp.com/video/JavaScript/File-Upload-Progress-Bar-Meter-Tutorial-Ajax-PHP
//use file_upload_parser.php from above url
ajax.setRequestHeader('Access-Control-Allow-Headers', '*');
ajax.setRequestHeader('Access-Control-Allow-Origin', '*');
ajax.send(formdata);
}
function progressHandler(event) {
_("loaded_n_total").innerHTML = "Uploaded " + event.loaded + " bytes of " + event.total;
var percent = Math.round((event.loaded / event.total) * 100);
_("progressBar").value = percent;
_("status").innerHTML = percent + "% uploaded... please wait";
}
function completeHandler(event) {
enableAll();
_("status").innerHTML = 'Upload Success!'; //event.target.responseText;
_("progressBar").value = 0; //wil clear progress bar after successful upload
}
function startHandler(event) {
disableAll();
}
function errorHandler(event) {
enableAll();
_("status").innerHTML = "Upload Failed";
}
function abortHandler(event) {
enableAll();
_("status").innerHTML = "Upload Aborted";
}
_('upload_form').addEventListener('submit', (e) => {
e.preventDefault();
uploadFile();
});
_('firmware-button').addEventListener('click',
function (e) {
e.target.classList.add('selected');
_('filesystem-button').classList.remove('selected');
_("file1").name = 'firmware';
}
)
_('filesystem-button').addEventListener('click',
function (e) {
e.target.classList.add('selected');
_('firmware-button').classList.remove('selected');
_("file1").name = 'filesystem';
}
)
_('file1').addEventListener('change', function(e){
var file = _("file1").files[0];
if (file && file.name){
_('button-send').disabled = false;
_("progressBar").value = 0; //wil clear progress bar after successful upload
}else{
_('button-send').disabled = true;
}
_('status').innerHTML = "Firmware Upload!";
_("loaded_n_total").innerHTML = "";
});
</script>
</div>
</body>
</html>
Code info
The code is quite simple, I use standard XMLHttpRequest to execute POST and I change the target (firmware or filesystem).
The form remain very simple.
<form enctype="multipart/form-data" class="form-container" id="upload_form" method="post">
<div id="switch-container">
<button class="button-o left selected" id="firmware-button" type="button">Firmware</button>
<button class="button-o right" id="filesystem-button" type="button">FileSystem</button>
</div>
<input accept='.bin,.bin.gz' class="input-file" id="file1" name='firmware' type="file"><br>
<progress id="progressBar" max="100" value="0"></progress>
<div class="additional-info">
<h3 id="status">Firmware upload</h3>
<p id="loaded_n_total"></p>
</div>
<hr/>
<button id="button-send" class="button-send" type="submit" disabled>Upload</button>
</form>
You can see that there isn’t the name of the input “filesystem” this because I manage It via JavaScript. The submit is stopped with this listener.
_('upload_form').addEventListener('submit', (e) => {
e.preventDefault();
uploadFile();
});
Then the uploadFile function is called.
function uploadFile() {
var file = _("file1").files[0];
var formdata = new FormData();
formdata.append(_("file1").name, file, file.name);
var ajax = new XMLHttpRequest();
ajax.upload.addEventListener("progress", progressHandler, false);
ajax.addEventListener("load", completeHandler, false);
ajax.addEventListener("loadstart", startHandler, false);
ajax.addEventListener("error", errorHandler, false);
ajax.addEventListener("abort", abortHandler, false);
ajax.open("POST", "/update");
// Enable CORS for testing
ajax.setRequestHeader('Access-Control-Allow-Headers', '*');
ajax.setRequestHeader('Access-Control-Allow-Origin', '*');
ajax.send(formdata);
}
I create new FormData and add only the file, than with the handlers
- “progress”: I manage scrollbar;
- “load”: manage the result;
- “loadstart”: I put an overlay to prevent clicking;
- “error” and “abort”: to manage errors.
Manage custom OTA page
We have 2 option to add this page to the sketch fully explained on this article “Web Server with esp8266 and esp32: byte array, gzipped pages and SPIFFS“, you can add it to the filesystem or as a bytearray. I use gzipped for all methods, you can check the simple version on linked article.
Add the page to filesystem
To upload a content to the filesystem you must install the relative plugin, select one of this
WeMos D1 mini (esp8266), integrated SPIFFS Filesystem (deprecated)
WeMos D1 mini (esp8266), integrated LittleFS Filesystem (reccomended)
So we are going to add data folder to the sketch and we put the gzipped page we want, now i put
index_color.html.gz
index_black_white.html.gz
for testing purpose. After that operation you can use the plugin like usual to upload the pages.
Sketch with LittleFS gzipped custom OTA page
Now we are going to change the WebUpdater.ino
file.
/*
* Custom OTA page from LittleFS file system
*
* Renzo Mischianti <www.mischianti.org>
*
* www.mischianti.org
*/
#include <ESP8266WiFi.h>
#include <WiFiClient.h>
#include <ESP8266WebServer.h>
#include <ESP8266mDNS.h>
#include <ESP8266HTTPUpdateServer.h>
#include <LittleFS.h>
#ifndef STASSID
#define STASSID "reef-casa-sopra"
#define STAPSK "aabbccdd77"
#endif
const char* host = "esp8266-webupdate";
const char* ssid = STASSID;
const char* password = STAPSK;
ESP8266WebServer httpServer(80);
ESP8266HTTPUpdateServer 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 (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");
}
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.");
}
Serial.print(F("Inizializing FS..."));
if (LittleFS.begin()){
Serial.println(F("done."));
}else{
Serial.println(F("fail."));
}
MDNS.begin(host);
// handler for the /update form page
httpServer.on("/update", HTTP_GET, handleRoot);
httpUpdater.setup(&httpServer);
httpServer.begin();
MDNS.addService("http", "tcp", 80);
Serial.printf("HTTPUpdateServer modified ready! Open http://%s.local/update in your browser\n", host);
}
void loop(void) {
httpServer.handleClient();
MDNS.update();
}
You can see the code explained in the linked article, and with
httpServer.on("/update", HTTP_GET, handleRoot);
I override the standard page on /update
GET, the handle
bool loadFromFS(String path, String dataType) {
Serial.print("Requested page -> ");
Serial.println(path);
if (LittleFS.exists(path)){
File dataFile = LittleFS.open(path, "r");
if (!dataFile) {
handleNotFound();
return false;
}
if (httpServer.streamFile(dataFile, dataType) != dataFile.size()) {
Serial.println("Sent less data than expected!");
}else{
Serial.println("Page served!");
}
dataFile.close();
}else{
handleNotFound();
return false;
}
return true;
}
void handleRoot() {
// loadFromFS("/index_color.html.gz", "text/html");
loadFromFS("/index_black_white.html.gz", "text/html");
}
read the selected page in handleRoot from LittleFS and return the stream.
Add custom page as bytearray
Get ready to use page
In the article linked before is explained in deep the bytearray page management, now in the repository of the pages you can grab the page with html extension,
https://github.com/xreef/ArduinoOTA_reusable_custom_web_page/minified
and put It in the bytearray converter by select the correct file (like index_color.html), and take the generated gzipped filearray.
Sketch with bytearray gzipped custom OTA page
Then we can go to modify the WebUpdater.ino
to response the selected page.
/*
* Custom OTA page in bytearray format
*
* Renzo Mischianti <www.mischianti.org>
*
* www.mischianti.org
*/
#include <ESP8266WiFi.h>
#include <WiFiClient.h>
#include <ESP8266WebServer.h>
#include <ESP8266mDNS.h>
#include <ESP8266HTTPUpdateServer.h>
#ifndef STASSID
#define STASSID "<YOUR-SSID>"
#define STAPSK "<YOUR-PASSWD>"
#endif
const char* host = "esp8266-webupdate";
const char* ssid = STASSID;
const char* password = STAPSK;
ESP8266WebServer httpServer(80);
ESP8266HTTPUpdateServer 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,
[...]
};
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);
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);
httpServer.begin();
MDNS.addService("http", "tcp", 80);
Serial.printf("HTTPUpdateServer modified ready! Open http://%s.local/update in your browser\n", host);
}
void loop(void) {
httpServer.handleClient();
MDNS.update();
}
You must add the complete bytearray of the page, in the code there is only a little part
the core code is very simple,
httpServer.on("/update", HTTP_GET, [&](){
httpServer.sendHeader(F("Content-Encoding"), F("gzip"));
httpServer.send_P(200, "text/html", (const char*)index_color_html_gz, (int)index_color_html_gz_len);
});
the send_P stream all the bytearray with Content-Encoding gzip, you can use the non gzipped bytearray but the size grow.
Thanks
- Firmware management
- OTA update with Arduino IDE
- OTA update with Web Browser
- Self OTA uptate from HTTP server
- Non standard Firmware update
I couldn’t get it to access page when using:
http://esp8266-webupdate.local/update
but it did work when using
192.168.1.xxx/update.
I used your example file from github.
Hi James,
as described on
esp8266 OTA update with Web Browser: firmware, filesystem and authentication – 1
you need an mDNS service on your PC or your router block that service.
Bye Renzo
How could you have the user set their own SSID and network password maybe in a text file without having to edit Arduino code and recompile? Would it be possible to do it via web browser interface?
Hi Steve,
yes It’s possible and not so difficult. In the project of Aurora Inverter Monitor I implement It.
I use a web interface to write a json file and store It in the internal flash, and ad reboot I check It, If exist I use that to configure the WiFi.
ABB Aurora Web Inverter Monitor (WIM): upload the sketch and front end – 9
Bye Renzo
Hello Renzo,
there seems to be a problem with your sketch. You change the original handle. The upload didn’t work for me. After changing the handle from “update” to “upload” and not overwriting the original handle, the upload worked. However, it immediately showed me 100% and after the upload time “Upload success”. But then nothing happens anymore, but the ESP boots properly.
I wrote a completely different update page for myself and call it up with “upload”. In the form I have set “action” to “update”. Then the page for waiting for the boot is called up automatically after the upload and then forwarded to the main page.
I downloaded your source code from Github. So I think that a typo is excluded.
greeting
Frank
Hi Frank,
I re-test all the process described in the tutorial with an WeMos D1 and the last core 3.1.2 and It’s work correctly for me.
Bye Renzo