mirror of
https://github.com/emsesp/EMS-ESP32.git
synced 2025-12-06 07:49:52 +03:00
278 lines
12 KiB
C++
278 lines
12 KiB
C++
/*
|
|
* EMS-ESP - https://github.com/emsesp/EMS-ESP
|
|
* Copyright 2020 Paul Derbyshire
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "emsesp.h"
|
|
|
|
namespace emsesp {
|
|
|
|
using namespace std::placeholders; // for `_1` etc
|
|
|
|
WebCustomizationService::WebCustomizationService(AsyncWebServer * server, FS * fs, SecurityManager * securityManager)
|
|
: _httpEndpoint(WebCustomization::read,
|
|
WebCustomization::update,
|
|
this,
|
|
server,
|
|
EMSESP_CUSTOMIZATION_SERVICE_PATH,
|
|
securityManager,
|
|
AuthenticationPredicates::IS_AUTHENTICATED)
|
|
, _fsPersistence(WebCustomization::read, WebCustomization::update, this, fs, EMSESP_CUSTOMIZATION_FILE)
|
|
, _masked_entities_handler(MASKED_ENTITIES_PATH,
|
|
securityManager->wrapCallback(std::bind(&WebCustomizationService::masked_entities, this, _1, _2),
|
|
AuthenticationPredicates::IS_AUTHENTICATED))
|
|
, _device_entities_handler(DEVICE_ENTITIES_PATH,
|
|
securityManager->wrapCallback(std::bind(&WebCustomizationService::device_entities, this, _1, _2),
|
|
AuthenticationPredicates::IS_AUTHENTICATED)) {
|
|
server->on(DEVICES_SERVICE_PATH,
|
|
HTTP_GET,
|
|
securityManager->wrapRequest(std::bind(&WebCustomizationService::devices, this, _1), AuthenticationPredicates::IS_AUTHENTICATED));
|
|
|
|
server->on(RESET_CUSTOMIZATION_SERVICE_PATH,
|
|
HTTP_POST,
|
|
securityManager->wrapRequest(std::bind(&WebCustomizationService::reset_customization, this, _1), AuthenticationPredicates::IS_ADMIN));
|
|
|
|
_masked_entities_handler.setMethod(HTTP_POST);
|
|
_masked_entities_handler.setMaxContentLength(2048);
|
|
_masked_entities_handler.setMaxJsonBufferSize(2048);
|
|
server->addHandler(&_masked_entities_handler);
|
|
|
|
_device_entities_handler.setMethod(HTTP_POST);
|
|
_device_entities_handler.setMaxContentLength(256);
|
|
server->addHandler(&_device_entities_handler);
|
|
}
|
|
|
|
// this creates the customization file, saving it to the FS
|
|
void WebCustomization::read(WebCustomization & settings, JsonObject & root) {
|
|
// Dallas Sensor customization
|
|
JsonArray sensorsJson = root.createNestedArray("sensors");
|
|
for (const SensorCustomization & sensor : settings.sensorCustomizations) {
|
|
JsonObject sensorJson = sensorsJson.createNestedObject();
|
|
sensorJson["id_str"] = sensor.id_str; // is
|
|
sensorJson["name"] = sensor.name; // n
|
|
sensorJson["offset"] = sensor.offset; // o
|
|
}
|
|
|
|
// Analog Sensor customization
|
|
JsonArray analogJson = root.createNestedArray("analogs");
|
|
for (const AnalogCustomization & sensor : settings.analogCustomizations) {
|
|
JsonObject sensorJson = analogJson.createNestedObject();
|
|
sensorJson["id"] = sensor.id; // i
|
|
sensorJson["name"] = sensor.name; // n
|
|
sensorJson["offset"] = sensor.offset; // o
|
|
sensorJson["factor"] = sensor.factor; // f
|
|
sensorJson["uom"] = sensor.uom; // u
|
|
sensorJson["type"] = sensor.type; // t
|
|
}
|
|
|
|
// Masked entities customization
|
|
JsonArray masked_entitiesJson = root.createNestedArray("masked_entities");
|
|
for (const EntityCustomization & entityCustomization : settings.entityCustomizations) {
|
|
JsonObject entityJson = masked_entitiesJson.createNestedObject();
|
|
entityJson["product_id"] = entityCustomization.product_id;
|
|
entityJson["device_id"] = entityCustomization.device_id;
|
|
|
|
JsonArray masked_entityJson = entityJson.createNestedArray("entity_ids");
|
|
for (std::string entity_id : entityCustomization.entity_ids) {
|
|
masked_entityJson.add(entity_id);
|
|
}
|
|
}
|
|
}
|
|
|
|
// call on initialization and also when the page is saved via web UI
|
|
// this loads the data into the internal class
|
|
StateUpdateResult WebCustomization::update(JsonObject & root, WebCustomization & settings) {
|
|
// Dallas Sensor customization
|
|
settings.sensorCustomizations.clear();
|
|
if (root["sensors"].is<JsonArray>()) {
|
|
for (const JsonObject sensorJson : root["sensors"].as<JsonArray>()) {
|
|
// create each of the sensor, overwritting any previous settings
|
|
auto sensor = SensorCustomization();
|
|
sensor.id_str = sensorJson["id_str"].as<std::string>();
|
|
sensor.name = sensorJson["name"].as<std::string>();
|
|
sensor.offset = sensorJson["offset"];
|
|
settings.sensorCustomizations.push_back(sensor); // add to list
|
|
}
|
|
}
|
|
|
|
// Analog Sensor customization
|
|
settings.analogCustomizations.clear();
|
|
if (root["analogs"].is<JsonArray>()) {
|
|
for (const JsonObject analogJson : root["analogs"].as<JsonArray>()) {
|
|
// create each of the sensor, overwritting any previous settings
|
|
auto sensor = AnalogCustomization();
|
|
sensor.id = analogJson["id"];
|
|
sensor.name = analogJson["name"].as<std::string>();
|
|
sensor.offset = analogJson["offset"];
|
|
sensor.factor = analogJson["factor"];
|
|
sensor.uom = analogJson["uom"];
|
|
sensor.type = analogJson["type"];
|
|
settings.analogCustomizations.push_back(sensor); // add to list
|
|
}
|
|
}
|
|
|
|
// load array of entities id's with masks, building up the object class
|
|
settings.entityCustomizations.clear();
|
|
if (root["masked_entities"].is<JsonArray>()) {
|
|
for (const JsonObject masked_entities : root["masked_entities"].as<JsonArray>()) {
|
|
auto new_entry = EntityCustomization();
|
|
new_entry.product_id = masked_entities["product_id"];
|
|
new_entry.device_id = masked_entities["device_id"];
|
|
|
|
for (const JsonVariant masked_entity_id : masked_entities["entity_ids"].as<JsonArray>()) {
|
|
if (masked_entity_id.is<std::string>()) {
|
|
new_entry.entity_ids.push_back(masked_entity_id.as<std::string>()); // add entity list
|
|
}
|
|
}
|
|
|
|
settings.entityCustomizations.push_back(new_entry); // save the new object
|
|
}
|
|
}
|
|
|
|
return StateUpdateResult::CHANGED;
|
|
}
|
|
|
|
// deletes the customization file
|
|
void WebCustomizationService::reset_customization(AsyncWebServerRequest * request) {
|
|
#ifndef EMSESP_STANDALONE
|
|
if (LITTLEFS.remove(EMSESP_CUSTOMIZATION_FILE)) {
|
|
AsyncWebServerResponse * response = request->beginResponse(200); // OK
|
|
request->send(response);
|
|
EMSESP::system_.restart_requested(true);
|
|
return;
|
|
}
|
|
// failed
|
|
AsyncWebServerResponse * response = request->beginResponse(204); // no content error
|
|
request->send(response);
|
|
#endif
|
|
}
|
|
|
|
// send back a list of devices used to the customization web page
|
|
void WebCustomizationService::devices(AsyncWebServerRequest * request) {
|
|
auto * response = new AsyncJsonResponse(false, EMSESP_JSON_SIZE_LARGE_DYN);
|
|
JsonObject root = response->getRoot();
|
|
|
|
JsonArray devices = root.createNestedArray("devices");
|
|
for (const auto & emsdevice : EMSESP::emsdevices) {
|
|
if (emsdevice->has_entities()) {
|
|
JsonObject obj = devices.createNestedObject();
|
|
obj["i"] = emsdevice->unique_id(); // a unique id
|
|
|
|
// shortname - we prefix the count to make it unique
|
|
uint8_t device_index = EMSESP::device_index(emsdevice->device_type(), emsdevice->unique_id());
|
|
if (device_index) {
|
|
char s[10];
|
|
obj["s"] = emsdevice->device_type_name() + Helpers::smallitoa(s, device_index);
|
|
} else {
|
|
obj["s"] = emsdevice->device_type_name();
|
|
}
|
|
}
|
|
}
|
|
|
|
response->setLength();
|
|
request->send(response);
|
|
}
|
|
|
|
// send back list of device entities
|
|
void WebCustomizationService::device_entities(AsyncWebServerRequest * request, JsonVariant & json) {
|
|
if (json.is<JsonObject>()) {
|
|
auto * response = new MsgpackAsyncJsonResponse(true, EMSESP_JSON_SIZE_XXXLARGE_DYN);
|
|
for (const auto & emsdevice : EMSESP::emsdevices) {
|
|
if (emsdevice->unique_id() == json["id"]) {
|
|
#ifndef EMSESP_STANDALONE
|
|
JsonArray output = response->getRoot();
|
|
emsdevice->generate_values_web_all(output);
|
|
#endif
|
|
response->setLength();
|
|
request->send(response);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// invalid, but send OK anyway
|
|
AsyncWebServerResponse * response = request->beginResponse(200);
|
|
request->send(response);
|
|
}
|
|
|
|
// takes a list of masked ids send from the webUI
|
|
// saves it in the customization service
|
|
// and updates the entity list real-time
|
|
void WebCustomizationService::masked_entities(AsyncWebServerRequest * request, JsonVariant & json) {
|
|
if (json.is<JsonObject>()) {
|
|
// find the device using the unique_id
|
|
for (const auto & emsdevice : EMSESP::emsdevices) {
|
|
if (emsdevice) {
|
|
uint8_t unique_device_id = json["id"];
|
|
if (emsdevice->unique_id() == unique_device_id) {
|
|
// first reset all the entity ids
|
|
emsdevice->reset_entity_masks();
|
|
|
|
// build a list of entities
|
|
JsonArray entity_ids_json = json["entity_ids"];
|
|
std::vector<std::string> entity_ids;
|
|
for (JsonVariant id : entity_ids_json) {
|
|
std::string entity_id = id.as<std::string>();
|
|
emsdevice->mask_entity(entity_id); // this will have immediate affect
|
|
entity_ids.push_back(entity_id);
|
|
}
|
|
|
|
// Save the list to the customization file
|
|
uint8_t product_id = emsdevice->product_id();
|
|
uint8_t device_id = emsdevice->device_id();
|
|
|
|
EMSESP::webCustomizationService.update(
|
|
[&](WebCustomization & settings) {
|
|
// if it exists (productid and deviceid match) overwrite it
|
|
for (auto & entityCustomization : settings.entityCustomizations) {
|
|
if ((entityCustomization.product_id == product_id) && (entityCustomization.device_id == device_id)) {
|
|
// already exists, clear the list and add the new values
|
|
entityCustomization.entity_ids.clear();
|
|
for (uint8_t i = 0; i < entity_ids.size(); i++) {
|
|
entityCustomization.entity_ids.push_back(entity_ids[i]);
|
|
}
|
|
return StateUpdateResult::CHANGED;
|
|
}
|
|
}
|
|
// create a new entry in the list
|
|
EntityCustomization new_entry;
|
|
new_entry.product_id = product_id;
|
|
new_entry.device_id = device_id;
|
|
for (uint8_t i = 0; i < entity_ids.size(); i++) {
|
|
new_entry.entity_ids.push_back(entity_ids[i]);
|
|
}
|
|
settings.entityCustomizations.push_back(new_entry);
|
|
return StateUpdateResult::CHANGED;
|
|
},
|
|
"local");
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
AsyncWebServerResponse * response = request->beginResponse(200); // OK
|
|
request->send(response);
|
|
}
|
|
|
|
// load the settings when the service starts
|
|
void WebCustomizationService::begin() {
|
|
_fsPersistence.readFromFS();
|
|
}
|
|
|
|
} // namespace emsesp
|