btclock_v3/src/lib/webserver.cpp

851 lines
26 KiB
C++
Raw Normal View History

2023-11-07 00:11:12 +00:00
#include "webserver.hpp"
AsyncWebServer server(80);
AsyncEventSource events("/events");
TaskHandle_t eventSourceTaskHandle;
2023-11-07 00:11:12 +00:00
2023-11-30 21:38:01 +00:00
void setupWebserver() {
events.onConnect([](AsyncEventSourceClient *client) {
client->send("welcome", NULL, millis(), 1000);
});
server.addHandler(&events);
2023-11-30 21:38:01 +00:00
// server.serveStatic("/css", LittleFS, "/css/");
// server.serveStatic("/js", LittleFS, "/js/");
server.serveStatic("/build", LittleFS, "/build");
server.serveStatic("/swagger.json", LittleFS, "/swagger.json");
server.serveStatic("/api.html", LittleFS, "/api.html");
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
server.on("/", HTTP_GET, onIndex);
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
server.on("/api/status", HTTP_GET, onApiStatus);
server.on("/api/system_status", HTTP_GET, onApiSystemStatus);
server.on("/api/wifi_set_tx_power", HTTP_GET, onApiSetWifiTxPower);
2023-11-21 15:12:44 +00:00
2023-11-30 21:38:01 +00:00
server.on("/api/full_refresh", HTTP_GET, onApiFullRefresh);
2023-11-07 20:26:15 +00:00
2024-01-31 22:45:26 +00:00
server.on("/api/stop_datasources", HTTP_GET, onApiStopDataSources);
server.on("/api/restart_datasources", HTTP_GET, onApiRestartDataSources);
2023-11-30 21:38:01 +00:00
server.on("/api/action/pause", HTTP_GET, onApiActionPause);
server.on("/api/action/timer_restart", HTTP_GET, onApiActionTimerRestart);
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
server.on("/api/settings", HTTP_GET, onApiSettingsGet);
server.on("/api/settings", HTTP_POST, onApiSettingsPost);
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
server.on("/api/show/screen", HTTP_GET, onApiShowScreen);
server.on("/api/show/text", HTTP_GET, onApiShowText);
2023-11-17 18:28:40 +00:00
2023-11-30 21:38:01 +00:00
AsyncCallbackJsonWebHandler *settingsPatchHandler =
new AsyncCallbackJsonWebHandler("/api/json/settings", onApiSettingsPatch);
server.addHandler(settingsPatchHandler);
2023-11-17 18:28:40 +00:00
2023-11-30 21:38:01 +00:00
AsyncCallbackJsonWebHandler *handler = new AsyncCallbackJsonWebHandler(
"/api/show/custom", onApiShowTextAdvanced);
server.addHandler(handler);
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
AsyncCallbackJsonWebHandler *lightsJsonHandler =
new AsyncCallbackJsonWebHandler("/api/lights", onApiLightsSetJson);
server.addHandler(lightsJsonHandler);
2023-11-30 21:38:01 +00:00
server.on("/api/lights/off", HTTP_GET, onApiLightsOff);
server.on("/api/lights/color", HTTP_GET, onApiLightsSetColor);
server.on("/api/lights", HTTP_GET, onApiLightsStatus);
2023-11-08 11:18:59 +00:00
2023-11-30 21:38:01 +00:00
// server.on("^\\/api\\/lights\\/([A-Fa-f0-9]{6}|[A-Fa-f0-9]{3})$", HTTP_GET,
// onApiLightsSetColor);
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
server.on("/api/restart", HTTP_GET, onApiRestart);
server.addRewrite(new OneParamRewrite("/api/lights/{color}",
"/api/lights/color?c={color}"));
server.addRewrite(
new OneParamRewrite("/api/show/screen/{s}", "/api/show/screen?s={s}"));
server.addRewrite(
new OneParamRewrite("/api/show/text/{text}", "/api/show/text?t={text}"));
server.addRewrite(new OneParamRewrite("/api/show/number/{number}",
"/api/show/text?t={text}"));
2023-11-07 00:11:12 +00:00
2023-11-30 21:38:01 +00:00
server.onNotFound(onNotFound);
2023-11-07 00:11:12 +00:00
2023-11-30 21:38:01 +00:00
DefaultHeaders::Instance().addHeader("Access-Control-Allow-Origin", "*");
DefaultHeaders::Instance().addHeader("Access-Control-Allow-Methods",
"GET, PATCH, POST, OPTIONS");
DefaultHeaders::Instance().addHeader("Access-Control-Allow-Headers", "*");
2023-11-30 21:38:01 +00:00
server.begin();
2023-11-30 21:38:01 +00:00
if (preferences.getBool("mdnsEnabled", true)) {
if (!MDNS.begin(getMyHostname())) {
Serial.println(F("Error setting up MDNS responder!"));
while (1) {
delay(1000);
}
}
2023-11-30 21:38:01 +00:00
MDNS.addService("http", "tcp", 80);
MDNS.addServiceTxt("http", "tcp", "model", "BTClock");
MDNS.addServiceTxt("http", "tcp", "version", "3.0");
MDNS.addServiceTxt("http", "tcp", "rev", GIT_REV);
}
2023-11-30 21:38:01 +00:00
xTaskCreate(eventSourceTask, "eventSourceTask", 4096, NULL, tskIDLE_PRIORITY,
&eventSourceTaskHandle);
2023-11-07 00:11:12 +00:00
}
2023-11-30 21:38:01 +00:00
void stopWebServer() { server.end(); }
2023-11-30 21:38:01 +00:00
StaticJsonDocument<768> getStatusObject() {
StaticJsonDocument<768> root;
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
root["currentScreen"] = getCurrentScreen();
root["numScreens"] = NUM_SCREENS;
root["timerRunning"] = isTimerActive();
root["espUptime"] = esp_timer_get_time() / 1000000;
// root["currentPrice"] = getPrice();
// root["currentBlockHeight"] = getBlockHeight();
root["espFreeHeap"] = ESP.getFreeHeap();
root["espHeapSize"] = ESP.getHeapSize();
// root["espFreePsram"] = ESP.getFreePsram();
// root["espPsramSize"] = ESP.getPsramSize();
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
JsonObject conStatus = root.createNestedObject("connectionStatus");
conStatus["price"] = isPriceNotifyConnected();
conStatus["blocks"] = isBlockNotifyConnected();
2023-11-08 11:18:59 +00:00
2023-11-30 21:38:01 +00:00
root["rssi"] = WiFi.RSSI();
2023-11-21 15:12:44 +00:00
2023-11-30 21:38:01 +00:00
return root;
}
2023-11-30 21:38:01 +00:00
StaticJsonDocument<512> getLedStatusObject() {
StaticJsonDocument<512> root;
JsonArray colors = root.createNestedArray("data");
// Adafruit_NeoPixel pix = getPixels();
for (uint i = 0; i < pixels.numPixels(); i++) {
uint32_t pixColor = pixels.getPixelColor(pixels.numPixels() - i - 1);
uint alpha = (pixColor >> 24) & 0xFF;
uint red = (pixColor >> 16) & 0xFF;
uint green = (pixColor >> 8) & 0xFF;
uint blue = pixColor & 0xFF;
char hexColor[8];
sprintf(hexColor, "#%02X%02X%02X", red, green, blue);
JsonObject object = colors.createNestedObject();
object["red"] = red;
object["green"] = green;
object["blue"] = blue;
object["hex"] = hexColor;
}
return root;
}
2023-11-30 21:38:01 +00:00
void eventSourceUpdate() {
2023-11-30 21:56:50 +00:00
if (!events.count()) return;
2023-11-30 21:38:01 +00:00
StaticJsonDocument<768> root = getStatusObject();
JsonArray data = root.createNestedArray("data");
2023-11-30 21:38:01 +00:00
root["leds"] = getLedStatusObject()["data"];
2023-11-30 21:38:01 +00:00
String epdContent[NUM_SCREENS];
std::array<String, NUM_SCREENS> retEpdContent = getCurrentEpdContent();
std::copy(std::begin(retEpdContent), std::end(retEpdContent), epdContent);
2023-11-30 21:38:01 +00:00
copyArray(epdContent, data);
2023-11-30 21:38:01 +00:00
String bufString;
serializeJson(root, bufString);
2023-11-30 21:38:01 +00:00
events.send(bufString.c_str(), "status");
}
/**
* @Api
* @Path("/api/status")
*/
2023-11-30 21:38:01 +00:00
void onApiStatus(AsyncWebServerRequest *request) {
AsyncResponseStream *response =
request->beginResponseStream("application/json");
2023-11-30 21:38:01 +00:00
StaticJsonDocument<1024> root = getStatusObject();
JsonArray data = root.createNestedArray("data");
JsonArray rendered = root.createNestedArray("rendered");
String epdContent[NUM_SCREENS];
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
root["leds"] = getLedStatusObject()["data"];
2023-11-19 01:23:47 +00:00
2023-11-30 21:38:01 +00:00
std::array<String, NUM_SCREENS> retEpdContent = getCurrentEpdContent();
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
std::copy(std::begin(retEpdContent), std::end(retEpdContent), epdContent);
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
copyArray(epdContent, data);
copyArray(epdContent, rendered);
serializeJson(root, *response);
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
request->send(response);
2023-11-07 20:26:15 +00:00
}
/**
* @Api
* @Path("/api/action/pause")
*/
2023-11-30 21:38:01 +00:00
void onApiActionPause(AsyncWebServerRequest *request) {
setTimerActive(false);
request->send(200);
2023-11-07 20:26:15 +00:00
};
/**
* @Api
* @Path("/api/action/timer_restart")
*/
2023-11-30 21:38:01 +00:00
void onApiActionTimerRestart(AsyncWebServerRequest *request) {
setTimerActive(true);
request->send(200);
2023-11-07 20:26:15 +00:00
}
/**
* @Api
* @Path("/api/full_refresh")
*/
2023-11-30 21:38:01 +00:00
void onApiFullRefresh(AsyncWebServerRequest *request) {
forceFullRefresh();
std::array<String, NUM_SCREENS> newEpdContent = getCurrentEpdContent();
2023-11-30 21:38:01 +00:00
setEpdContent(newEpdContent, true);
2023-11-30 21:38:01 +00:00
request->send(200);
}
2023-11-14 22:09:23 +00:00
/**
* @Api
* @Path("/api/show/screen")
*/
2023-11-30 21:38:01 +00:00
void onApiShowScreen(AsyncWebServerRequest *request) {
if (request->hasParam("s")) {
AsyncWebParameter *p = request->getParam("s");
uint currentScreen = p->value().toInt();
setCurrentScreen(currentScreen);
}
request->send(200);
2023-11-07 20:26:15 +00:00
}
2023-11-30 21:38:01 +00:00
void onApiShowText(AsyncWebServerRequest *request) {
if (request->hasParam("t")) {
AsyncWebParameter *p = request->getParam("t");
String t = p->value();
2023-11-30 21:56:50 +00:00
t.toUpperCase(); // This is needed as long as lowercase letters are glitchy
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
std::array<String, NUM_SCREENS> textEpdContent;
for (uint i = 0; i < NUM_SCREENS; i++) {
textEpdContent[i] = t[i];
2023-11-14 18:46:29 +00:00
}
2023-11-30 21:38:01 +00:00
setEpdContent(textEpdContent);
}
setCurrentScreen(SCREEN_CUSTOM);
request->send(200);
2023-11-14 18:46:29 +00:00
}
2023-11-30 21:38:01 +00:00
void onApiShowTextAdvanced(AsyncWebServerRequest *request, JsonVariant &json) {
JsonArray screens = json.as<JsonArray>();
2023-11-22 15:26:57 +00:00
2023-11-30 21:38:01 +00:00
std::array<String, NUM_SCREENS> epdContent;
int i = 0;
for (JsonVariant s : screens) {
epdContent[i] = s.as<String>();
i++;
}
2023-11-22 15:26:57 +00:00
2023-11-30 21:38:01 +00:00
setEpdContent(epdContent);
2023-11-17 18:28:40 +00:00
2023-11-30 21:38:01 +00:00
setCurrentScreen(SCREEN_CUSTOM);
request->send(200);
}
2023-11-17 18:28:40 +00:00
2023-11-30 21:38:01 +00:00
void onApiSettingsPatch(AsyncWebServerRequest *request, JsonVariant &json) {
JsonObject settings = json.as<JsonObject>();
bool settingsChanged = true;
if (settings.containsKey("fgColor")) {
String fgColor = settings["fgColor"].as<String>();
preferences.putUInt("fgColor", strtol(fgColor.c_str(), NULL, 16));
setFgColor(int(strtol(fgColor.c_str(), NULL, 16)));
Serial.print(F("Setting foreground color to "));
Serial.println(strtol(fgColor.c_str(), NULL, 16));
settingsChanged = true;
}
if (settings.containsKey("bgColor")) {
String bgColor = settings["bgColor"].as<String>();
preferences.putUInt("bgColor", strtol(bgColor.c_str(), NULL, 16));
setBgColor(int(strtol(bgColor.c_str(), NULL, 16)));
Serial.print(F("Setting background color to "));
Serial.println(bgColor.c_str());
settingsChanged = true;
}
if (settings.containsKey("timePerScreen")) {
preferences.putUInt("timerSeconds",
settings["timePerScreen"].as<uint>() * 60);
}
String strSettings[] = {"hostnamePrefix", "mempoolInstance"};
for (String setting : strSettings) {
if (settings.containsKey(setting)) {
preferences.putString(setting.c_str(), settings[setting].as<String>());
Serial.printf("Setting %s to %s\r\n", setting.c_str(),
settings[setting].as<String>());
}
}
String uintSettings[] = {"minSecPriceUpd", "fullRefreshMin", "ledBrightness"};
for (String setting : uintSettings) {
if (settings.containsKey(setting)) {
preferences.putUInt(setting.c_str(), settings[setting].as<uint>());
Serial.printf("Setting %s to %d\r\n", setting.c_str(),
settings[setting].as<uint>());
}
}
if (settings.containsKey("tzOffset")) {
int gmtOffset = settings["tzOffset"].as<int>() * 60;
size_t written = preferences.putInt("gmtOffset", gmtOffset);
Serial.printf("Setting %s to %d (%d minutes, written %d)\r\n", "gmtOffset",
gmtOffset, settings["tzOffset"].as<int>(), written);
}
String boolSettings[] = {"fetchEurPrice", "ledTestOnPower", "ledFlashOnUpd",
"mdnsEnabled", "otaEnabled", "stealFocus",
"mcapBigChar"};
for (String setting : boolSettings) {
if (settings.containsKey(setting)) {
preferences.putBool(setting.c_str(), settings[setting].as<boolean>());
Serial.printf("Setting %s to %d\r\n", setting.c_str(),
settings[setting].as<boolean>());
}
}
if (settings.containsKey("screens")) {
for (JsonVariant screen : settings["screens"].as<JsonArray>()) {
JsonObject s = screen.as<JsonObject>();
uint id = s["id"].as<uint>();
String key = "screen[" + String(id) + "]";
String prefKey = "screen" + String(id) + "Visible";
bool visible = s["enabled"].as<boolean>();
preferences.putBool(prefKey.c_str(), visible);
}
}
if (settings.containsKey("txPower")) {
int txPower = settings["txPower"].as<int>();
if (txPower == 80) {
preferences.remove("txPower");
if (WiFi.getTxPower() != 80) {
ESP.restart();
}
} else if (static_cast<int>(wifi_power_t::WIFI_POWER_MINUS_1dBm) <=
txPower &&
txPower <= static_cast<int>(wifi_power_t::WIFI_POWER_19_5dBm)) {
// is valid value
if (WiFi.setTxPower(static_cast<wifi_power_t>(txPower))) {
Serial.printf("Set WiFi Tx power to: %d\n", txPower);
preferences.putInt("txPower", txPower);
settingsChanged = true;
}
2023-11-21 15:12:44 +00:00
}
2023-11-30 21:38:01 +00:00
}
2023-11-21 15:12:44 +00:00
2023-11-30 21:38:01 +00:00
request->send(200);
if (settingsChanged) {
queueLedEffect(LED_FLASH_SUCCESS);
}
2023-11-17 18:28:40 +00:00
}
2023-11-30 21:38:01 +00:00
void onApiRestart(AsyncWebServerRequest *request) {
request->send(200);
2023-11-30 21:56:50 +00:00
if (events.count()) events.send("closing");
2023-11-30 21:38:01 +00:00
delay(500);
2023-11-30 21:38:01 +00:00
esp_restart();
2023-11-07 20:26:15 +00:00
}
/**
* @Api
* @Method GET
* @Path("/api/settings")
*/
2023-11-30 21:38:01 +00:00
void onApiSettingsGet(AsyncWebServerRequest *request) {
StaticJsonDocument<1536> root;
root["numScreens"] = NUM_SCREENS;
root["fgColor"] = getFgColor();
root["bgColor"] = getBgColor();
root["timerSeconds"] = getTimerSeconds();
root["timerRunning"] = isTimerActive();
root["minSecPriceUpd"] = preferences.getUInt(
"minSecPriceUpd", DEFAULT_SECONDS_BETWEEN_PRICE_UPDATE);
root["fullRefreshMin"] =
preferences.getUInt("fullRefreshMin", DEFAULT_MINUTES_FULL_REFRESH);
root["wpTimeout"] = preferences.getUInt("wpTimeout", 600);
root["tzOffset"] = preferences.getInt("gmtOffset", TIME_OFFSET_SECONDS) / 60;
root["useBitcoinNode"] = preferences.getBool("useNode", false);
root["mempoolInstance"] =
preferences.getString("mempoolInstance", DEFAULT_MEMPOOL_INSTANCE);
root["ledTestOnPower"] = preferences.getBool("ledTestOnPower", true);
root["ledFlashOnUpd"] = preferences.getBool("ledFlashOnUpd", false);
root["ledBrightness"] = preferences.getUInt("ledBrightness", 128);
root["stealFocus"] = preferences.getBool("stealFocus", true);
root["mcapBigChar"] = preferences.getBool("mcapBigChar", true);
root["mdnsEnabled"] = preferences.getBool("mdnsEnabled", true);
root["otaEnabled"] = preferences.getBool("otaEnabled", true);
root["fetchEurPrice"] = preferences.getBool("fetchEurPrice", false);
root["hostnamePrefix"] = preferences.getString("hostnamePrefix", "btclock");
root["hostname"] = getMyHostname();
root["ip"] = WiFi.localIP();
root["txPower"] = WiFi.getTxPower();
2023-11-07 20:26:15 +00:00
#ifdef GIT_REV
2023-11-30 21:38:01 +00:00
root["gitRev"] = String(GIT_REV);
2023-11-07 20:26:15 +00:00
#endif
#ifdef LAST_BUILD_TIME
2023-11-30 21:38:01 +00:00
root["lastBuildTime"] = String(LAST_BUILD_TIME);
2023-11-07 20:26:15 +00:00
#endif
2023-11-30 21:38:01 +00:00
JsonArray screens = root.createNestedArray("screens");
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
std::vector<std::string> screenNameMap = getScreenNameMap();
2023-11-08 11:18:59 +00:00
2023-11-30 21:38:01 +00:00
for (int i = 0; i < screenNameMap.size(); i++) {
JsonObject o = screens.createNestedObject();
String key = "screen" + String(i) + "Visible";
o["id"] = i;
o["name"] = screenNameMap[i];
o["enabled"] = preferences.getBool(key.c_str(), true);
}
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
AsyncResponseStream *response =
request->beginResponseStream("application/json");
serializeJson(root, *response);
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
request->send(response);
2023-11-07 20:26:15 +00:00
}
2023-11-30 21:38:01 +00:00
bool processEpdColorSettings(AsyncWebServerRequest *request) {
bool settingsChanged = false;
if (request->hasParam("fgColor", true)) {
AsyncWebParameter *fgColor = request->getParam("fgColor", true);
preferences.putUInt("fgColor", strtol(fgColor->value().c_str(), NULL, 16));
setFgColor(int(strtol(fgColor->value().c_str(), NULL, 16)));
// Serial.print(F("Setting foreground color to "));
// Serial.println(fgColor->value().c_str());
settingsChanged = true;
}
if (request->hasParam("bgColor", true)) {
AsyncWebParameter *bgColor = request->getParam("bgColor", true);
preferences.putUInt("bgColor", strtol(bgColor->value().c_str(), NULL, 16));
setBgColor(int(strtol(bgColor->value().c_str(), NULL, 16)));
// Serial.print(F("Setting background color to "));
// Serial.println(bgColor->value().c_str());
settingsChanged = true;
}
return settingsChanged;
2023-11-07 20:26:15 +00:00
}
2023-11-30 21:38:01 +00:00
void onApiSettingsPost(AsyncWebServerRequest *request) {
bool settingsChanged = false;
settingsChanged = processEpdColorSettings(request);
int headers = request->headers();
int i;
for (i = 0; i < headers; i++) {
AsyncWebHeader *h = request->getHeader(i);
Serial.printf("HEADER[%s]: %s\n", h->name().c_str(), h->value().c_str());
}
int params = request->params();
for (int i = 0; i < params; i++) {
AsyncWebParameter *p = request->getParam(i);
2023-11-30 21:56:50 +00:00
if (p->isFile()) { // p->isPost() is also true
2023-11-30 21:38:01 +00:00
Serial.printf("FILE[%s]: %s, size: %u\n", p->name().c_str(),
p->value().c_str(), p->size());
} else if (p->isPost()) {
Serial.printf("POST[%s]: %s\n", p->name().c_str(), p->value().c_str());
} else {
Serial.printf("GET[%s]: %s\n", p->name().c_str(), p->value().c_str());
}
}
if (request->hasParam("fetchEurPrice", true)) {
AsyncWebParameter *fetchEurPrice = request->getParam("fetchEurPrice", true);
preferences.putBool("fetchEurPrice", fetchEurPrice->value().toInt());
settingsChanged = true;
} else {
preferences.putBool("fetchEurPrice", 0);
settingsChanged = true;
}
if (request->hasParam("ledTestOnPower", true)) {
AsyncWebParameter *ledTestOnPower =
request->getParam("ledTestOnPower", true);
preferences.putBool("ledTestOnPower", ledTestOnPower->value().toInt());
settingsChanged = true;
} else {
preferences.putBool("ledTestOnPower", 0);
settingsChanged = true;
}
if (request->hasParam("ledFlashOnUpd", true)) {
AsyncWebParameter *ledFlashOnUpdate =
request->getParam("ledFlashOnUpd", true);
preferences.putBool("ledFlashOnUpd", ledFlashOnUpdate->value().toInt());
settingsChanged = true;
} else {
preferences.putBool("ledFlashOnUpd", 0);
settingsChanged = true;
}
if (request->hasParam("mdnsEnabled", true)) {
AsyncWebParameter *mdnsEnabled = request->getParam("mdnsEnabled", true);
preferences.putBool("mdnsEnabled", mdnsEnabled->value().toInt());
settingsChanged = true;
} else {
preferences.putBool("mdnsEnabled", 0);
settingsChanged = true;
}
if (request->hasParam("otaEnabled", true)) {
AsyncWebParameter *otaEnabled = request->getParam("otaEnabled", true);
preferences.putBool("otaEnabled", otaEnabled->value().toInt());
settingsChanged = true;
} else {
preferences.putBool("otaEnabled", 0);
settingsChanged = true;
}
if (request->hasParam("stealFocusOnBlock", true)) {
AsyncWebParameter *stealFocusOnBlock =
request->getParam("stealFocusOnBlock", true);
preferences.putBool("stealFocus", stealFocusOnBlock->value().toInt());
settingsChanged = true;
} else {
preferences.putBool("stealFocus", 0);
settingsChanged = true;
}
if (request->hasParam("mcapBigChar", true)) {
AsyncWebParameter *mcapBigChar = request->getParam("mcapBigChar", true);
preferences.putBool("mcapBigChar", mcapBigChar->value().toInt());
settingsChanged = true;
} else {
preferences.putBool("mcapBigChar", 0);
settingsChanged = true;
}
if (request->hasParam("mempoolInstance", true)) {
AsyncWebParameter *mempoolInstance =
request->getParam("mempoolInstance", true);
preferences.putString("mempoolInstance", mempoolInstance->value().c_str());
settingsChanged = true;
}
if (request->hasParam("hostnamePrefix", true)) {
AsyncWebParameter *hostnamePrefix =
request->getParam("hostnamePrefix", true);
preferences.putString("hostnamePrefix", hostnamePrefix->value().c_str());
settingsChanged = true;
}
if (request->hasParam("ledBrightness", true)) {
AsyncWebParameter *ledBrightness = request->getParam("ledBrightness", true);
preferences.putUInt("ledBrightness", ledBrightness->value().toInt());
settingsChanged = true;
}
if (request->hasParam("fullRefreshMin", true)) {
AsyncWebParameter *fullRefreshMin =
request->getParam("fullRefreshMin", true);
preferences.putUInt("fullRefreshMin", fullRefreshMin->value().toInt());
settingsChanged = true;
}
if (request->hasParam("wpTimeout", true)) {
AsyncWebParameter *wpTimeout = request->getParam("wpTimeout", true);
preferences.putUInt("wpTimeout", wpTimeout->value().toInt());
settingsChanged = true;
}
std::vector<std::string> screenNameMap = getScreenNameMap();
if (request->hasParam("screens")) {
AsyncWebParameter *screenParam = request->getParam("screens", true);
Serial.printf(screenParam->value().c_str());
}
for (int i = 0; i < screenNameMap.size(); i++) {
String key = "screen[" + String(i) + "]";
String prefKey = "screen" + String(i) + "Visible";
bool visible = false;
if (request->hasParam(key, true)) {
AsyncWebParameter *screenParam = request->getParam(key, true);
visible = screenParam->value().toInt();
}
preferences.putBool(prefKey.c_str(), visible);
}
if (request->hasParam("tzOffset", true)) {
AsyncWebParameter *p = request->getParam("tzOffset", true);
int tzOffsetSeconds = p->value().toInt() * 60;
preferences.putInt("gmtOffset", tzOffsetSeconds);
settingsChanged = true;
}
if (request->hasParam("minSecPriceUpd", true)) {
AsyncWebParameter *p = request->getParam("minSecPriceUpd", true);
int minSecPriceUpd = p->value().toInt();
preferences.putUInt("minSecPriceUpd", minSecPriceUpd);
settingsChanged = true;
}
if (request->hasParam("timePerScreen", true)) {
AsyncWebParameter *p = request->getParam("timePerScreen", true);
uint timerSeconds = p->value().toInt() * 60;
preferences.putUInt("timerSeconds", timerSeconds);
settingsChanged = true;
}
request->send(200);
if (settingsChanged) {
queueLedEffect(LED_FLASH_SUCCESS);
}
}
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
void onApiSystemStatus(AsyncWebServerRequest *request) {
AsyncResponseStream *response =
request->beginResponseStream("application/json");
2023-11-08 11:18:59 +00:00
2023-11-30 21:38:01 +00:00
StaticJsonDocument<128> root;
2023-11-17 18:28:40 +00:00
2023-11-30 21:38:01 +00:00
root["espFreeHeap"] = ESP.getFreeHeap();
root["espHeapSize"] = ESP.getHeapSize();
root["espFreePsram"] = ESP.getFreePsram();
root["espPsramSize"] = ESP.getPsramSize();
root["rssi"] = WiFi.RSSI();
root["txPower"] = WiFi.getTxPower();
2023-11-17 18:28:40 +00:00
2023-11-30 21:38:01 +00:00
serializeJson(root, *response);
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
request->send(response);
}
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
#define STRINGIFY(x) #x
#define ENUM_TO_STRING(x) STRINGIFY(x)
2023-11-30 21:38:01 +00:00
void onApiSetWifiTxPower(AsyncWebServerRequest *request) {
if (request->hasParam("txPower")) {
AsyncWebParameter *txPowerParam = request->getParam("txPower");
int txPower = txPowerParam->value().toInt();
if (static_cast<int>(wifi_power_t::WIFI_POWER_MINUS_1dBm) <= txPower &&
txPower <= static_cast<int>(wifi_power_t::WIFI_POWER_19_5dBm)) {
// is valid value
String txPowerName =
std::to_string(
static_cast<std::underlying_type_t<wifi_power_t>>(txPower))
.c_str();
Serial.printf("Set WiFi Tx power to: %s\n", txPowerName);
if (WiFi.setTxPower(static_cast<wifi_power_t>(txPower))) {
preferences.putInt("txPower", txPower);
request->send(200, "application/json", "{\"setTxPower\": \"ok\"}");
return;
}
2023-11-07 20:26:15 +00:00
}
2023-11-30 21:38:01 +00:00
}
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
return request->send(400);
2023-11-07 20:26:15 +00:00
}
2023-11-30 21:38:01 +00:00
void onApiLightsStatus(AsyncWebServerRequest *request) {
AsyncResponseStream *response =
request->beginResponseStream("application/json");
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
serializeJson(getLedStatusObject()["data"], *response);
2023-11-07 20:26:15 +00:00
2023-11-30 21:38:01 +00:00
request->send(response);
}
2023-11-07 20:26:15 +00:00
2024-01-31 22:45:26 +00:00
void onApiStopDataSources(AsyncWebServerRequest *request) {
AsyncResponseStream *response =
request->beginResponseStream("application/json");
stopPriceNotify();
stopBlockNotify();
request->send(response);
}
void onApiRestartDataSources(AsyncWebServerRequest *request) {
AsyncResponseStream *response =
request->beginResponseStream("application/json");
stopPriceNotify();
stopBlockNotify();
setupPriceNotify();
setupBlockNotify();
request->send(response);
}
2023-11-30 21:38:01 +00:00
void onApiLightsOff(AsyncWebServerRequest *request) {
setLights(0, 0, 0);
request->send(200);
2023-11-07 20:26:15 +00:00
}
2023-11-30 21:38:01 +00:00
void onApiLightsSetColor(AsyncWebServerRequest *request) {
if (request->hasParam("c")) {
AsyncResponseStream *response =
request->beginResponseStream("application/json");
2023-11-21 15:12:44 +00:00
2023-11-30 21:38:01 +00:00
String rgbColor = request->getParam("c")->value();
2023-11-21 15:12:44 +00:00
2023-11-30 21:38:01 +00:00
if (rgbColor.compareTo("off") == 0) {
setLights(0, 0, 0);
} else {
uint r, g, b;
sscanf(rgbColor.c_str(), "%02x%02x%02x", &r, &g, &b);
setLights(r, g, b);
2023-11-21 15:12:44 +00:00
}
2023-11-30 21:38:01 +00:00
StaticJsonDocument<48> doc;
doc["result"] = rgbColor;
serializeJson(getLedStatusObject()["data"], *response);
request->send(response);
2023-11-30 21:38:01 +00:00
} else {
request->send(400);
}
}
2023-11-30 21:38:01 +00:00
void onApiLightsSetJson(AsyncWebServerRequest *request, JsonVariant &json) {
JsonArray lights = json.as<JsonArray>();
2023-11-19 01:23:47 +00:00
2023-11-30 21:38:01 +00:00
if (lights.size() != pixels.numPixels()) {
Serial.printf("Invalid values for LED set %d\n", lights.size());
request->send(400);
return;
}
2023-11-19 01:23:47 +00:00
2023-11-30 21:38:01 +00:00
for (uint i = 0; i < pixels.numPixels(); i++) {
unsigned int red, green, blue;
2023-11-08 11:18:59 +00:00
2023-11-30 21:38:01 +00:00
if (lights[i].containsKey("red") && lights[i].containsKey("green") &&
lights[i].containsKey("blue")) {
red = lights[i]["red"].as<uint>();
green = lights[i]["green"].as<uint>();
blue = lights[i]["blue"].as<uint>();
} else if (lights[i].containsKey("hex")) {
if (!sscanf(lights[i]["hex"].as<String>().c_str(), "#%02X%02X%02X", &red,
&green, &blue) == 3) {
Serial.printf("Invalid hex for LED %d\n", i);
2023-11-18 13:47:45 +00:00
request->send(400);
return;
2023-11-30 21:38:01 +00:00
}
} else {
Serial.printf("No valid color for LED %d\n", i);
request->send(400);
return;
2023-11-18 13:47:45 +00:00
}
2023-11-30 21:38:01 +00:00
pixels.setPixelColor((pixels.numPixels() - i - 1),
pixels.Color(red, green, blue));
}
2023-11-18 13:47:45 +00:00
2023-11-30 21:38:01 +00:00
pixels.show();
saveLedState();
2023-11-18 13:47:45 +00:00
2023-11-30 21:38:01 +00:00
request->send(200);
}
2023-11-30 21:38:01 +00:00
void onIndex(AsyncWebServerRequest *request) {
request->send(LittleFS, "/index.html", String(), false);
}
void onNotFound(AsyncWebServerRequest *request) {
// Serial.printf("NotFound, URL[%s]\n", request->url());
// Serial.printf("NotFound, METHOD[%s]\n", request->methodToString());
// int headers = request->headers();
// int i;
// for (i = 0; i < headers; i++)
// {
// AsyncWebHeader *h = request->getHeader(i);
// Serial.printf("NotFound HEADER[%s]: %s\n", h->name().c_str(),
// h->value().c_str());
// }
// int params = request->params();
// for (int i = 0; i < params; i++)
// {
// AsyncWebParameter *p = request->getParam(i);
// if (p->isFile())
// { // p->isPost() is also true
// Serial.printf("NotFound FILE[%s]: %s, size: %u\n",
// p->name().c_str(), p->value().c_str(), p->size());
// }
// else if (p->isPost())
// {
// Serial.printf("NotFound POST[%s]: %s\n", p->name().c_str(),
// p->value().c_str());
// }
// else
// {
// Serial.printf("NotFound GET[%s]: %s\n", p->name().c_str(),
// p->value().c_str());
// }
// }
// Access-Control-Request-Method == POST might be better
if (request->method() == HTTP_OPTIONS ||
request->hasHeader("Sec-Fetch-Mode")) {
// Serial.printf("NotFound, Return[%d]\n", 200);
request->send(200);
} else {
// Serial.printf("NotFound, Return[%d]\n", 404);
request->send(404);
}
2023-11-07 00:11:12 +00:00
};
2023-11-30 21:38:01 +00:00
void eventSourceTask(void *pvParameters) {
for (;;) {
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
eventSourceUpdate();
}
}