WLED/wled00/wled18_server.ino

346 lines
11 KiB
Arduino
Raw Normal View History

/*
* Server page definitions
*/
2019-09-05 22:45:59 +02:00
//Is this an IP?
bool isIp(String str) {
for (size_t i = 0; i < str.length(); i++) {
int c = str.charAt(i);
if (c != '.' && (c < '0' || c > '9')) {
return false;
}
}
return true;
}
bool captivePortal(AsyncWebServerRequest *request)
{
2019-09-12 12:40:06 +02:00
if (ON_STA_FILTER(request)) return false; //only serve captive in AP mode
2019-09-05 22:45:59 +02:00
String hostH;
if (!request->hasHeader("Host")) return false;
hostH = request->getHeader("Host")->value();
if (!isIp(hostH) && hostH.indexOf("wled.me") < 0 && hostH.indexOf(cmDNS) < 0) {
DEBUG_PRINTLN("Captive portal");
AsyncWebServerResponse *response = request->beginResponse(302);
2019-09-12 12:40:06 +02:00
response->addHeader("Location", "http://4.3.2.1");
2019-09-05 22:45:59 +02:00
request->send(response);
return true;
}
return false;
}
void initServer()
{
2019-02-14 17:25:41 +01:00
//CORS compatiblity
DefaultHeaders::Instance().addHeader("Access-Control-Allow-Origin", "*");
DefaultHeaders::Instance().addHeader("Access-Control-Allow-Methods", "*");
DefaultHeaders::Instance().addHeader("Access-Control-Allow-Headers", "*");
2019-11-30 19:17:25 +01:00
server.on("/liveview", HTTP_GET, [](AsyncWebServerRequest *request){
request->send_P(200, "text/html", PAGE_liveview);
});
2019-02-14 17:25:41 +01:00
//settings page
2019-02-16 00:21:22 +01:00
server.on("/settings", HTTP_GET, [](AsyncWebServerRequest *request){
serveSettings(request);
});
2019-02-16 00:21:22 +01:00
server.on("/favicon.ico", HTTP_GET, [](AsyncWebServerRequest *request){
2019-03-16 02:09:37 +01:00
if(!handleFileRead(request, "/favicon.ico"))
{
2019-02-16 00:21:22 +01:00
request->send_P(200, "image/x-icon", favicon, 156);
}
});
2019-02-16 00:21:22 +01:00
server.on("/sliders", HTTP_GET, [](AsyncWebServerRequest *request){
serveIndex(request);
});
2019-02-16 00:21:22 +01:00
server.on("/welcome", HTTP_GET, [](AsyncWebServerRequest *request){
serveSettings(request);
});
2019-02-16 00:21:22 +01:00
server.on("/reset", HTTP_GET, [](AsyncWebServerRequest *request){
serveMessage(request, 200,"Rebooting now...","Please wait ~10 seconds...",129);
doReboot = true;
});
2019-02-16 00:21:22 +01:00
server.on("/settings/wifi", HTTP_POST, [](AsyncWebServerRequest *request){
if (!(wifiLock && otaLock)) handleSettingsSet(request, 1);
serveMessage(request, 200,"WiFi settings saved.","Please connect to the new IP (if changed)",129);
forceReconnect = true;
});
2019-02-16 00:21:22 +01:00
server.on("/settings/leds", HTTP_POST, [](AsyncWebServerRequest *request){
handleSettingsSet(request, 2);
serveMessage(request, 200,"LED settings saved.","Redirecting...",1);
});
2019-02-16 00:21:22 +01:00
server.on("/settings/ui", HTTP_POST, [](AsyncWebServerRequest *request){
handleSettingsSet(request, 3);
serveMessage(request, 200,"UI settings saved.","Redirecting...",1);
});
2019-02-16 00:21:22 +01:00
server.on("/settings/sync", HTTP_POST, [](AsyncWebServerRequest *request){
handleSettingsSet(request, 4);
2019-02-18 22:34:21 +01:00
serveMessage(request, 200,"Sync settings saved.","Redirecting...",1);
});
2019-02-16 00:21:22 +01:00
server.on("/settings/time", HTTP_POST, [](AsyncWebServerRequest *request){
handleSettingsSet(request, 5);
serveMessage(request, 200,"Time settings saved.","Redirecting...",1);
});
2019-02-16 00:21:22 +01:00
server.on("/settings/sec", HTTP_POST, [](AsyncWebServerRequest *request){
handleSettingsSet(request, 6);
if (!doReboot) serveMessage(request, 200,"Security settings saved.","Rebooting, please wait ~10 seconds...",129);
doReboot = true;
});
2019-02-10 23:05:06 +01:00
2019-03-06 01:20:38 +01:00
server.on("/json", HTTP_GET, [](AsyncWebServerRequest *request){
serveJson(request);
});
2019-12-18 00:41:45 +01:00
AsyncCallbackJsonWebHandler* handler = new AsyncCallbackJsonWebHandler("/json", [](AsyncWebServerRequest *request) {
bool verboseResponse = false;
2019-12-21 03:17:54 +01:00
{ //scope JsonDocument so it releases its buffer
2019-12-18 00:41:45 +01:00
DynamicJsonDocument jsonBuffer(8192);
DeserializationError error = deserializeJson(jsonBuffer, (uint8_t*)(request->_tempObject));
JsonObject root = jsonBuffer.as<JsonObject>();
if (error || root.isNull()) {
request->send(400, "application/json", "{\"error\":10}"); return;
}
verboseResponse = deserializeState(root);
}
if (verboseResponse) { //if JSON contains "v"
serveJson(request); return;
}
2019-03-06 01:20:38 +01:00
request->send(200, "application/json", "{\"success\":true}");
});
server.addHandler(handler);
2019-02-21 16:32:15 +01:00
//*******DEPRECATED*******
2019-02-16 00:21:22 +01:00
server.on("/version", HTTP_GET, [](AsyncWebServerRequest *request){
2019-02-14 17:25:41 +01:00
request->send(200, "text/plain", (String)VERSION);
});
2019-02-16 00:21:22 +01:00
server.on("/uptime", HTTP_GET, [](AsyncWebServerRequest *request){
2019-02-14 17:25:41 +01:00
request->send(200, "text/plain", (String)millis());
});
2019-02-16 00:21:22 +01:00
server.on("/freeheap", HTTP_GET, [](AsyncWebServerRequest *request){
2019-02-14 17:25:41 +01:00
request->send(200, "text/plain", (String)ESP.getFreeHeap());
});
2019-02-21 16:32:15 +01:00
//*******END*******/
2019-02-16 00:21:22 +01:00
server.on("/u", HTTP_GET, [](AsyncWebServerRequest *request){
request->send_P(200, "text/html", PAGE_usermod);
});
2019-02-16 00:21:22 +01:00
server.on("/teapot", HTTP_GET, [](AsyncWebServerRequest *request){
serveMessage(request, 418, "418. I'm a teapot.", "(Tangible Embedded Advanced Project Of Twinkling)", 254);
});
//if OTA is allowed
if (!otaLock){
2019-03-16 02:09:37 +01:00
#if !defined WLED_DISABLE_FILESYSTEM && defined WLED_ENABLE_FS_EDITOR
#ifdef ARDUINO_ARCH_ESP32
server.addHandler(new SPIFFSEditor(SPIFFS));//http_username,http_password));
#else
server.addHandler(new SPIFFSEditor());//http_username,http_password));
#endif
#else
2019-02-16 00:21:22 +01:00
server.on("/edit", HTTP_GET, [](AsyncWebServerRequest *request){
2019-03-16 02:09:37 +01:00
serveMessage(request, 501, "Not implemented", "The SPIFFS editor is disabled in this build.", 254);
});
#endif
//init ota page
#ifndef WLED_DISABLE_OTA
server.on("/update", HTTP_GET, [](AsyncWebServerRequest *request){
2019-12-02 12:41:35 +01:00
request->send_P(200, "text/html", PAGE_update);
});
server.on("/update", HTTP_POST, [](AsyncWebServerRequest *request){
if (Update.hasError())
{
serveMessage(request, 500, "Failed updating firmware!", "Please check your file and retry!", 254); return;
}
serveMessage(request, 200, "Successfully updated firmware!", "Please wait while the module reboots...", 131);
doReboot = true;
},[](AsyncWebServerRequest *request, String filename, size_t index, uint8_t *data, size_t len, bool final){
if(!index){
DEBUG_PRINTLN("OTA Update Start");
#ifdef ESP8266
Update.runAsync(true);
#endif
Update.begin((ESP.getFreeSketchSpace() - 0x1000) & 0xFFFFF000);
}
if(!Update.hasError()) Update.write(data, len);
if(final){
if(Update.end(true)){
DEBUG_PRINTLN("Update Success");
} else {
DEBUG_PRINTLN("Update Failed");
}
}
});
#else
2019-02-16 00:21:22 +01:00
server.on("/update", HTTP_GET, [](AsyncWebServerRequest *request){
2019-03-16 02:09:37 +01:00
serveMessage(request, 501, "Not implemented", "OTA updates are disabled in this build.", 254);
});
#endif
} else
{
2019-02-16 00:21:22 +01:00
server.on("/edit", HTTP_GET, [](AsyncWebServerRequest *request){
2019-03-16 02:09:37 +01:00
serveMessage(request, 500, "Access Denied", "Please unlock OTA in security settings!", 254);
});
2019-02-16 00:21:22 +01:00
server.on("/update", HTTP_GET, [](AsyncWebServerRequest *request){
2019-03-16 02:09:37 +01:00
serveMessage(request, 500, "Access Denied", "Please unlock OTA in security settings!", 254);
});
}
2019-02-16 00:21:22 +01:00
server.on("/", HTTP_GET, [](AsyncWebServerRequest *request){
2019-09-05 22:45:59 +02:00
if (captivePortal(request)) return;
2019-02-16 00:21:22 +01:00
serveIndexOrWelcome(request);
});
//called when the url is not defined here, ajax-in; get-settings
2019-02-16 00:21:22 +01:00
server.onNotFound([](AsyncWebServerRequest *request){
DEBUG_PRINTLN("Not-Found HTTP call:");
2019-02-18 22:34:21 +01:00
DEBUG_PRINTLN("URI: " + request->url());
2019-09-05 22:45:59 +02:00
if (captivePortal(request)) return;
//make API CORS compatible
2019-02-16 00:21:22 +01:00
if (request->method() == HTTP_OPTIONS)
{
2019-02-16 00:21:22 +01:00
request->send(200); return;
}
2019-03-16 02:09:37 +01:00
if(handleSet(request, request->url())) return;
#ifndef WLED_DISABLE_ALEXA
if(espalexa.handleAlexaApiCall(request)) return;
#endif
#ifdef WLED_ENABLE_FS_SERVING
if(handleFileRead(request, request->url())) return;
#endif
request->send(404, "text/plain", "Not Found");
});
}
2019-02-16 00:21:22 +01:00
void serveIndexOrWelcome(AsyncWebServerRequest *request)
{
if (!showWelcomePage){
2019-02-16 00:21:22 +01:00
serveIndex(request);
} else {
serveSettings(request);
}
}
2019-02-16 00:21:22 +01:00
void serveIndex(AsyncWebServerRequest* request)
{
2019-03-16 02:09:37 +01:00
#ifdef WLED_ENABLE_FS_SERVING
2019-12-02 12:41:35 +01:00
if (handleFileRead(request, "/index.htm")) return;
2019-03-16 02:09:37 +01:00
#endif
2019-12-02 12:41:35 +01:00
AsyncWebServerResponse *response = request->beginResponse_P(200, "text/html", PAGE_index, PAGE_index_L);
2019-02-16 00:21:22 +01:00
response->addHeader("Content-Encoding","gzip");
2019-02-16 00:21:22 +01:00
request->send(response);
}
2019-02-16 00:21:22 +01:00
String msgProcessor(const String& var)
{
if (var == "MSG") {
2019-03-09 21:41:23 +01:00
String messageBody = messageHead;
2019-02-16 00:21:22 +01:00
messageBody += "</h2>";
messageBody += messageSub;
uint32_t optt = optionType;
2019-02-16 00:21:22 +01:00
if (optt < 60) //redirect to settings after optionType seconds
2019-02-16 00:21:22 +01:00
{
messageBody += "<script>setTimeout(RS," + String(optt*1000) + ")</script>";
} else if (optt < 120) //redirect back after optionType-60 seconds, unused
2019-02-16 00:21:22 +01:00
{
//messageBody += "<script>setTimeout(B," + String((optt-60)*1000) + ")</script>";
} else if (optt < 180) //reload parent after optionType-120 seconds
2019-02-16 00:21:22 +01:00
{
messageBody += "<script>setTimeout(RP," + String((optt-120)*1000) + ")</script>";
} else if (optt == 253)
2019-02-16 00:21:22 +01:00
{
messageBody += "<br><br><form action=/settings><button class=\"bt\" type=submit>Back</button></form>"; //button to settings
} else if (optt == 254)
2019-02-16 00:21:22 +01:00
{
messageBody += "<br><br><button type=\"button\" class=\"bt\" onclick=\"B()\">Back</button>";
2019-02-16 00:21:22 +01:00
}
return messageBody;
}
return String();
}
void serveMessage(AsyncWebServerRequest* request, uint16_t code, String headl, String subl="", byte optionT=255)
{
2019-02-16 00:21:22 +01:00
messageHead = headl;
messageSub = subl;
optionType = optionT;
2019-02-16 00:21:22 +01:00
request->send_P(code, "text/html", PAGE_msg, msgProcessor);
}
String settingsProcessor(const String& var)
{
2019-03-11 19:30:49 +01:00
if (var == "CSS") {
2019-03-16 02:09:37 +01:00
char buf[2048];
getSettingsJS(optionType, buf);
return String(buf);
2019-03-11 19:30:49 +01:00
}
if (var == "SCSS") return String(FPSTR(PAGE_settingsCss));
2019-02-16 00:21:22 +01:00
return String();
}
2019-02-16 00:21:22 +01:00
void serveSettings(AsyncWebServerRequest* request)
{
2019-02-16 00:21:22 +01:00
byte subPage = 0;
2019-03-06 01:20:38 +01:00
const String& url = request->url();
2019-02-16 00:21:22 +01:00
if (url.indexOf("sett") >= 0)
{
if (url.indexOf("wifi") > 0) subPage = 1;
else if (url.indexOf("leds") > 0) subPage = 2;
else if (url.indexOf("ui") > 0) subPage = 3;
else if (url.indexOf("sync") > 0) subPage = 4;
else if (url.indexOf("time") > 0) subPage = 5;
else if (url.indexOf("sec") > 0) subPage = 6;
} else subPage = 255; //welcome page
if (subPage == 1 && wifiLock && otaLock)
{
2019-03-05 10:59:15 +01:00
serveMessage(request, 500, "Access Denied", "Please unlock OTA in security settings!", 254); return;
2019-02-16 00:21:22 +01:00
}
#ifdef WLED_DISABLE_MOBILE_UI //disable welcome page if not enough storage
2019-02-16 00:21:22 +01:00
if (subPage == 255) {serveIndex(request); return;}
#endif
2019-03-11 19:30:49 +01:00
optionType = subPage;
switch (subPage)
{
2019-02-16 00:21:22 +01:00
case 1: request->send_P(200, "text/html", PAGE_settings_wifi, settingsProcessor); break;
case 2: request->send_P(200, "text/html", PAGE_settings_leds, settingsProcessor); break;
case 3: request->send_P(200, "text/html", PAGE_settings_ui , settingsProcessor); break;
case 4: request->send_P(200, "text/html", PAGE_settings_sync, settingsProcessor); break;
case 5: request->send_P(200, "text/html", PAGE_settings_time, settingsProcessor); break;
case 6: request->send_P(200, "text/html", PAGE_settings_sec , settingsProcessor); break;
2019-09-12 12:40:06 +02:00
case 255: request->send_P(200, "text/html", PAGE_welcome); break;
2019-12-02 12:41:35 +01:00
default: request->send_P(200, "text/html", PAGE_settings);
}
}