mirror of
https://github.com/esp8266/Arduino.git
synced 2025-04-19 23:22:16 +03:00
Merge pull request #913 from me-no-dev/esp8266
Add SPIFFS WebServer Example
This commit is contained in:
commit
0772a69b1c
239
libraries/ESP8266WebServer/examples/FSBrowser/FSBrowser.ino
Normal file
239
libraries/ESP8266WebServer/examples/FSBrowser/FSBrowser.ino
Normal file
@ -0,0 +1,239 @@
|
|||||||
|
/*
|
||||||
|
FSWebServer - Example WebServer with SPIFFS backend for esp8266
|
||||||
|
Copyright (c) 2015 Hristo Gochkov. All rights reserved.
|
||||||
|
This file is part of the ESP8266WebServer library for Arduino environment.
|
||||||
|
|
||||||
|
This library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Lesser General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 2.1 of the License, or (at your option) any later version.
|
||||||
|
This library is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with this library; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
|
||||||
|
upload the contents of the data folder with MkSPIFFS Tool ("ESP8266 Sketch Data Upload" in Tools menu in Arduino IDE)
|
||||||
|
or you can upload the contents of a folder if you CD in that folder and run the following command:
|
||||||
|
for file in `ls -A1`; do curl -F "file=@$PWD/$file" esp8266fs.local/edit; done
|
||||||
|
|
||||||
|
access the sample web page at http://esp8266fs.local
|
||||||
|
edit the page by going to http://esp8266fs.local/edit
|
||||||
|
*/
|
||||||
|
#include <ESP8266WiFi.h>
|
||||||
|
#include <WiFiClient.h>
|
||||||
|
#include <ESP8266WebServer.h>
|
||||||
|
#include <ESP8266mDNS.h>
|
||||||
|
#include <FS.h>
|
||||||
|
|
||||||
|
#define DBG_OUTPUT_PORT Serial
|
||||||
|
|
||||||
|
const char* ssid = "wifi-ssid";
|
||||||
|
const char* password = "wifi-password";
|
||||||
|
const char* host = "esp8266fs";
|
||||||
|
|
||||||
|
ESP8266WebServer server(80);
|
||||||
|
//holds the current upload
|
||||||
|
File fsUploadFile;
|
||||||
|
|
||||||
|
//format bytes
|
||||||
|
String formatBytes(size_t bytes){
|
||||||
|
if (bytes < 1024){
|
||||||
|
return String(bytes)+"B";
|
||||||
|
} else if(bytes < (1024 * 1024)){
|
||||||
|
return String(bytes/1024.0)+"KB";
|
||||||
|
} else if(bytes < (1024 * 1024 * 1024)){
|
||||||
|
return String(bytes/1024.0/1024.0)+"MB";
|
||||||
|
} else {
|
||||||
|
return String(bytes/1024.0/1024.0/1024.0)+"GB";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
String getContentType(String filename){
|
||||||
|
if(server.hasArg("download")) return "application/octet-stream";
|
||||||
|
else if(filename.endsWith(".htm")) return "text/html";
|
||||||
|
else if(filename.endsWith(".html")) return "text/html";
|
||||||
|
else if(filename.endsWith(".css")) return "text/css";
|
||||||
|
else if(filename.endsWith(".js")) return "application/javascript";
|
||||||
|
else if(filename.endsWith(".png")) return "image/png";
|
||||||
|
else if(filename.endsWith(".gif")) return "image/gif";
|
||||||
|
else if(filename.endsWith(".jpg")) return "image/jpeg";
|
||||||
|
else if(filename.endsWith(".ico")) return "image/x-icon";
|
||||||
|
else if(filename.endsWith(".xml")) return "text/xml";
|
||||||
|
else if(filename.endsWith(".pdf")) return "application/x-pdf";
|
||||||
|
else if(filename.endsWith(".zip")) return "application/x-zip";
|
||||||
|
else if(filename.endsWith(".gz")) return "application/x-gzip";
|
||||||
|
return "text/plain";
|
||||||
|
}
|
||||||
|
|
||||||
|
bool handleFileRead(String path){
|
||||||
|
DBG_OUTPUT_PORT.println("handleFileRead: " + path);
|
||||||
|
if(path.endsWith("/")) path += "index.htm";
|
||||||
|
String contentType = getContentType(path);
|
||||||
|
String pathWithGz = path + ".gz";
|
||||||
|
if(SPIFFS.exists(pathWithGz) || SPIFFS.exists(path)){
|
||||||
|
if(SPIFFS.exists(pathWithGz))
|
||||||
|
path += ".gz";
|
||||||
|
File file = SPIFFS.open(path, "r");
|
||||||
|
size_t sent = server.streamFile(file, contentType);
|
||||||
|
file.close();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void handleFileUpload(){
|
||||||
|
if(server.uri() != "/edit") return;
|
||||||
|
HTTPUpload& upload = server.upload();
|
||||||
|
if(upload.status == UPLOAD_FILE_START){
|
||||||
|
String filename = upload.filename;
|
||||||
|
if(!filename.startsWith("/")) filename = "/"+filename;
|
||||||
|
DBG_OUTPUT_PORT.print("handleFileUpload Name: "); DBG_OUTPUT_PORT.println(filename);
|
||||||
|
fsUploadFile = SPIFFS.open(filename, "w");
|
||||||
|
filename = String();
|
||||||
|
} else if(upload.status == UPLOAD_FILE_WRITE){
|
||||||
|
//DBG_OUTPUT_PORT.print("handleFileUpload Data: "); DBG_OUTPUT_PORT.println(upload.currentSize);
|
||||||
|
if(fsUploadFile)
|
||||||
|
fsUploadFile.write(upload.buf, upload.currentSize);
|
||||||
|
} else if(upload.status == UPLOAD_FILE_END){
|
||||||
|
if(fsUploadFile)
|
||||||
|
fsUploadFile.close();
|
||||||
|
DBG_OUTPUT_PORT.print("handleFileUpload Size: "); DBG_OUTPUT_PORT.println(upload.totalSize);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void handleFileDelete(){
|
||||||
|
if(server.args() == 0) return server.send(500, "text/plain", "BAD ARGS");
|
||||||
|
String path = server.arg(0);
|
||||||
|
DBG_OUTPUT_PORT.println("handleFileDelete: " + path);
|
||||||
|
if(path == "/")
|
||||||
|
return server.send(500, "text/plain", "BAD PATH");
|
||||||
|
if(!SPIFFS.exists(path))
|
||||||
|
return server.send(404, "text/plain", "FileNotFound");
|
||||||
|
SPIFFS.remove(path);
|
||||||
|
server.send(200, "text/plain", "");
|
||||||
|
path = String();
|
||||||
|
}
|
||||||
|
|
||||||
|
void handleFileCreate(){
|
||||||
|
if(server.args() == 0)
|
||||||
|
return server.send(500, "text/plain", "BAD ARGS");
|
||||||
|
String path = server.arg(0);
|
||||||
|
DBG_OUTPUT_PORT.println("handleFileCreate: " + path);
|
||||||
|
if(path == "/")
|
||||||
|
return server.send(500, "text/plain", "BAD PATH");
|
||||||
|
if(SPIFFS.exists(path))
|
||||||
|
return server.send(500, "text/plain", "FILE EXISTS");
|
||||||
|
File file = SPIFFS.open(path, "w");
|
||||||
|
if(file)
|
||||||
|
file.close();
|
||||||
|
else
|
||||||
|
return server.send(500, "text/plain", "CREATE FAILED");
|
||||||
|
server.send(200, "text/plain", "");
|
||||||
|
path = String();
|
||||||
|
}
|
||||||
|
|
||||||
|
void handleFileList() {
|
||||||
|
if(!server.hasArg("dir")) {server.send(500, "text/plain", "BAD ARGS"); return;}
|
||||||
|
|
||||||
|
String path = server.arg("dir");
|
||||||
|
DBG_OUTPUT_PORT.println("handleFileList: " + path);
|
||||||
|
Dir dir = SPIFFS.openDir(path);
|
||||||
|
path = String();
|
||||||
|
|
||||||
|
String output = "[";
|
||||||
|
while(dir.next()){
|
||||||
|
File entry = dir.openFile("r");
|
||||||
|
if (output != "[") output += ',';
|
||||||
|
bool isDir = false;
|
||||||
|
output += "{\"type\":\"";
|
||||||
|
output += (isDir)?"dir":"file";
|
||||||
|
output += "\",\"name\":\"";
|
||||||
|
output += String(entry.name()).substring(1);
|
||||||
|
output += "\"}";
|
||||||
|
entry.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
output += "]";
|
||||||
|
server.send(200, "text/json", output);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setup(void){
|
||||||
|
DBG_OUTPUT_PORT.begin(115200);
|
||||||
|
DBG_OUTPUT_PORT.print("\n");
|
||||||
|
DBG_OUTPUT_PORT.setDebugOutput(true);
|
||||||
|
SPIFFS.begin();
|
||||||
|
{
|
||||||
|
Dir dir = SPIFFS.openDir("/");
|
||||||
|
while (dir.next()) {
|
||||||
|
String fileName = dir.fileName();
|
||||||
|
size_t fileSize = dir.fileSize();
|
||||||
|
DBG_OUTPUT_PORT.printf("FS File: %s, size: %s\n", fileName.c_str(), formatBytes(fileSize).c_str());
|
||||||
|
}
|
||||||
|
DBG_OUTPUT_PORT.printf("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//WIFI INIT
|
||||||
|
DBG_OUTPUT_PORT.printf("Connecting to %s\n", ssid);
|
||||||
|
if (String(WiFi.SSID()) != String(ssid)) {
|
||||||
|
WiFi.begin(ssid, password);
|
||||||
|
}
|
||||||
|
|
||||||
|
while (WiFi.status() != WL_CONNECTED) {
|
||||||
|
delay(500);
|
||||||
|
DBG_OUTPUT_PORT.print(".");
|
||||||
|
}
|
||||||
|
DBG_OUTPUT_PORT.println("");
|
||||||
|
DBG_OUTPUT_PORT.print("Connected! IP address: ");
|
||||||
|
DBG_OUTPUT_PORT.println(WiFi.localIP());
|
||||||
|
|
||||||
|
MDNS.begin(host);
|
||||||
|
DBG_OUTPUT_PORT.print("Open http://");
|
||||||
|
DBG_OUTPUT_PORT.print(host);
|
||||||
|
DBG_OUTPUT_PORT.println(".local/edit to see the file browser");
|
||||||
|
|
||||||
|
|
||||||
|
//SERVER INIT
|
||||||
|
//list directory
|
||||||
|
server.on("/list", HTTP_GET, handleFileList);
|
||||||
|
//load editor
|
||||||
|
server.on("/edit", HTTP_GET, [](){
|
||||||
|
if(!handleFileRead("/edit.htm")) server.send(404, "text/plain", "FileNotFound");
|
||||||
|
});
|
||||||
|
//create file
|
||||||
|
server.on("/edit", HTTP_PUT, handleFileCreate);
|
||||||
|
//delete file
|
||||||
|
server.on("/edit", HTTP_DELETE, handleFileDelete);
|
||||||
|
//called after file upload
|
||||||
|
server.on("/edit", HTTP_POST, [](){ server.send(200, "text/plain", ""); });
|
||||||
|
//called when a file is received inside POST data
|
||||||
|
server.onFileUpload(handleFileUpload);
|
||||||
|
|
||||||
|
//called when the url is not defined here
|
||||||
|
//use it to load content from SPIFFS
|
||||||
|
server.onNotFound([](){
|
||||||
|
if(!handleFileRead(server.uri()))
|
||||||
|
server.send(404, "text/plain", "FileNotFound");
|
||||||
|
});
|
||||||
|
|
||||||
|
//get heap status, analog input value and all GPIO statuses in one json call
|
||||||
|
server.on("/all", HTTP_GET, [](){
|
||||||
|
String json = "{";
|
||||||
|
json += "\"heap\":"+String(ESP.getFreeHeap());
|
||||||
|
json += ", \"analog\":"+String(analogRead(A0));
|
||||||
|
json += ", \"gpio\":"+String((uint32_t)(((GPI | GPO) & 0xFFFF) | ((GP16I & 0x01) << 16)));
|
||||||
|
json += "}";
|
||||||
|
server.send(200, "text/json", json);
|
||||||
|
json = String();
|
||||||
|
});
|
||||||
|
server.begin();
|
||||||
|
DBG_OUTPUT_PORT.println("HTTP server started");
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void loop(void){
|
||||||
|
server.handleClient();
|
||||||
|
}
|
BIN
libraries/ESP8266WebServer/examples/FSBrowser/data/edit.htm.gz
Normal file
BIN
libraries/ESP8266WebServer/examples/FSBrowser/data/edit.htm.gz
Normal file
Binary file not shown.
BIN
libraries/ESP8266WebServer/examples/FSBrowser/data/favicon.ico
Normal file
BIN
libraries/ESP8266WebServer/examples/FSBrowser/data/favicon.ico
Normal file
Binary file not shown.
After Width: | Height: | Size: 1.1 KiB |
BIN
libraries/ESP8266WebServer/examples/FSBrowser/data/graphs.js.gz
Normal file
BIN
libraries/ESP8266WebServer/examples/FSBrowser/data/graphs.js.gz
Normal file
Binary file not shown.
97
libraries/ESP8266WebServer/examples/FSBrowser/data/index.htm
Normal file
97
libraries/ESP8266WebServer/examples/FSBrowser/data/index.htm
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
<!--
|
||||||
|
FSWebServer - Example Index Page
|
||||||
|
Copyright (c) 2015 Hristo Gochkov. All rights reserved.
|
||||||
|
This file is part of the ESP8266WebServer library for Arduino environment.
|
||||||
|
|
||||||
|
This library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Lesser General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 2.1 of the License, or (at your option) any later version.
|
||||||
|
This library is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with this library; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
-->
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta http-equiv="Content-type" content="text/html; charset=utf-8">
|
||||||
|
<title>ESP Monitor</title>
|
||||||
|
<script type="text/javascript" src="graphs.js"></script>
|
||||||
|
<script type="text/javascript">
|
||||||
|
var heap,temp,digi;
|
||||||
|
var reloadPeriod = 1000;
|
||||||
|
var running = false;
|
||||||
|
|
||||||
|
function loadValues(){
|
||||||
|
if(!running) return;
|
||||||
|
var xh = new XMLHttpRequest();
|
||||||
|
xh.onreadystatechange = function(){
|
||||||
|
if (xh.readyState == 4){
|
||||||
|
if(xh.status == 200) {
|
||||||
|
var res = JSON.parse(xh.responseText);
|
||||||
|
heap.add(res.heap);
|
||||||
|
temp.add(res.analog);
|
||||||
|
digi.add(res.gpio);
|
||||||
|
if(running) setTimeout(loadValues, reloadPeriod);
|
||||||
|
} else running = false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
xh.open("GET", "/all", true);
|
||||||
|
xh.send(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
function run(){
|
||||||
|
if(!running){
|
||||||
|
running = true;
|
||||||
|
loadValues();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onBodyLoad(){
|
||||||
|
var refreshInput = document.getElementById("refresh-rate");
|
||||||
|
refreshInput.value = reloadPeriod;
|
||||||
|
refreshInput.onchange = function(e){
|
||||||
|
var value = parseInt(e.target.value);
|
||||||
|
reloadPeriod = (value > 0)?value:0;
|
||||||
|
e.target.value = reloadPeriod;
|
||||||
|
}
|
||||||
|
var stopButton = document.getElementById("stop-button");
|
||||||
|
stopButton.onclick = function(e){
|
||||||
|
running = false;
|
||||||
|
}
|
||||||
|
var startButton = document.getElementById("start-button");
|
||||||
|
startButton.onclick = function(e){
|
||||||
|
run();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Example with 10K thermistor
|
||||||
|
//function calcThermistor(v) {
|
||||||
|
// var t = Math.log(((10230000 / v) - 10000));
|
||||||
|
// t = (1/(0.001129148+(0.000234125*t)+(0.0000000876741*t*t*t)))-273.15;
|
||||||
|
// return (t>120)?0:Math.round(t*10)/10;
|
||||||
|
//}
|
||||||
|
//temp = createGraph(document.getElementById("analog"), "Temperature", 100, 128, 10, 40, false, "cyan", calcThermistor);
|
||||||
|
|
||||||
|
temp = createGraph(document.getElementById("analog"), "Analog Input", 100, 128, 0, 1023, false, "cyan");
|
||||||
|
heap = createGraph(document.getElementById("heap"), "Current Heap", 100, 125, 0, 30000, true, "orange");
|
||||||
|
digi = createDigiGraph(document.getElementById("digital"), "GPIO", 100, 146, [0, 4, 5, 16], "gold");
|
||||||
|
run();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</head>
|
||||||
|
<body id="index" style="margin:0; padding:0;" onload="onBodyLoad()">
|
||||||
|
<div id="controls" style="display: block; border: 1px solid rgb(68, 68, 68); padding: 5px; margin: 5px; width: 362px; background-color: rgb(238, 238, 238);">
|
||||||
|
<label>Period (ms):</label>
|
||||||
|
<input type="number" id="refresh-rate"/>
|
||||||
|
<input type="button" id="start-button" value="Start"/>
|
||||||
|
<input type="button" id="stop-button" value="Stop"/>
|
||||||
|
</div>
|
||||||
|
<div id="heap"></div>
|
||||||
|
<div id="analog"></div>
|
||||||
|
<div id="digital"></div>
|
||||||
|
</body>
|
||||||
|
</html>
|
Loading…
x
Reference in New Issue
Block a user