/* * 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 . */ #include "emsesp.h" namespace emsesp { using namespace std::placeholders; // for `_1` etc WebDataService::WebDataService(AsyncWebServer * server, SecurityManager * securityManager) : _device_dataHandler(DEVICE_DATA_SERVICE_PATH, securityManager->wrapCallback(std::bind(&WebDataService::device_data, this, _1, _2), AuthenticationPredicates::IS_AUTHENTICATED)) , _writevalue_dataHandler(WRITE_VALUE_SERVICE_PATH, securityManager->wrapCallback(std::bind(&WebDataService::write_value, this, _1, _2), AuthenticationPredicates::IS_ADMIN)) , _writesensor_dataHandler(WRITE_SENSOR_SERVICE_PATH, securityManager->wrapCallback(std::bind(&WebDataService::write_sensor, this, _1, _2), AuthenticationPredicates::IS_ADMIN)) { server->on(EMSESP_DATA_SERVICE_PATH, HTTP_GET, securityManager->wrapRequest(std::bind(&WebDataService::all_devices, this, _1), AuthenticationPredicates::IS_AUTHENTICATED)); server->on(SCAN_DEVICES_SERVICE_PATH, HTTP_GET, securityManager->wrapRequest(std::bind(&WebDataService::scan_devices, this, _1), AuthenticationPredicates::IS_AUTHENTICATED)); _device_dataHandler.setMethod(HTTP_POST); _device_dataHandler.setMaxContentLength(256); server->addHandler(&_device_dataHandler); _writevalue_dataHandler.setMethod(HTTP_POST); _writevalue_dataHandler.setMaxContentLength(256); server->addHandler(&_writevalue_dataHandler); _writesensor_dataHandler.setMethod(HTTP_POST); _writesensor_dataHandler.setMaxContentLength(256); server->addHandler(&_writesensor_dataHandler); } void WebDataService::scan_devices(AsyncWebServerRequest * request) { EMSESP::scan_devices(); request->send(200); } void WebDataService::all_devices(AsyncWebServerRequest * request) { AsyncJsonResponse * 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) { JsonObject obj = devices.createNestedObject(); obj["id"] = emsdevice->unique_id(); obj["type"] = emsdevice->device_type_name(); obj["brand"] = emsdevice->brand_to_string(); obj["name"] = emsdevice->name(); obj["deviceid"] = emsdevice->device_id(); obj["productid"] = emsdevice->product_id(); obj["version"] = emsdevice->version(); } } JsonArray sensors = root.createNestedArray("sensors"); if (EMSESP::have_sensors()) { uint8_t i = 1; for (const auto & sensor : EMSESP::sensor_devices()) { JsonObject obj = sensors.createNestedObject(); obj["no"] = i++; obj["id"] = sensor.to_string(true); obj["temp"] = (float)(sensor.temperature_c) / 10; obj["offset"] = (float)(sensor.offset()) / 10; } } if (EMSESP::system_.analog_enabled()) { root["analog"] = EMSESP::system_.analog(); } response->setLength(); request->send(response); } // The unique_id is the unique record ID from the Web table to identify which device to load // Compresses the JSON using MsgPack https://msgpack.org/index.html void WebDataService::device_data(AsyncWebServerRequest * request, JsonVariant & json) { if (json.is()) { MsgpackAsyncJsonResponse * response = new MsgpackAsyncJsonResponse(false, EMSESP_JSON_SIZE_XXLARGE_DYN); for (const auto & emsdevice : EMSESP::emsdevices) { if (emsdevice) { if (emsdevice->unique_id() == json["id"]) { #ifndef EMSESP_STANDALONE JsonObject root = response->getRoot(); emsdevice->generate_values_json_web(root); #endif response->setLength(); request->send(response); return; } } } } // invalid AsyncWebServerResponse * response = request->beginResponse(200); request->send(response); } // takes a command and its data value from a specific Device, from the Web // assumes the service has been checked for admin authentication void WebDataService::write_value(AsyncWebServerRequest * request, JsonVariant & json) { if (json.is()) { JsonObject dv = json["devicevalue"]; uint8_t id = json["id"]; // using the unique ID from the web find the real device type for (const auto & emsdevice : EMSESP::emsdevices) { if (emsdevice) { if (emsdevice->unique_id() == id) { const char * cmd = dv["c"]; uint8_t device_type = emsdevice->device_type(); uint8_t cmd_return = 1; // OK char s[10]; // the data could be in any format, but we need string JsonVariant data = dv["v"]; if (data.is()) { cmd_return = Command::call(device_type, cmd, data.as(), true); } else if (data.is()) { cmd_return = Command::call(device_type, cmd, Helpers::render_value(s, data.as(), 0), true); } else if (data.is()) { cmd_return = Command::call(device_type, cmd, Helpers::render_value(s, (float)data.as(), 1), true); } else if (data.is()) { cmd_return = Command::call(device_type, cmd, data.as() ? "true" : "false", true); } // send "Write command sent to device" or "Write command failed" AsyncWebServerResponse * response = request->beginResponse((cmd_return == 1) ? 200 : 204); request->send(response); return; } } } } AsyncWebServerResponse * response = request->beginResponse(204); // Write command failed request->send(response); } // takes a sensorname and optional offset from the Web void WebDataService::write_sensor(AsyncWebServerRequest * request, JsonVariant & json) { bool ok = false; if (json.is()) { JsonObject sensor = json["sensor"]; // if valid add. uint8_t no = sensor["no"]; if (no > 0 && no < 100) { char name[25]; std::string id = sensor["id"]; strlcpy(name, id.c_str(), sizeof(name)); float offset = sensor["offset"]; // this will be a float value. We'll convert it to int and * 10 it int16_t offset10 = offset * 10; char idstr[3]; ok = EMSESP::dallassensor_.update(Helpers::itoa(idstr, no, 10), name, offset10); } } AsyncWebServerResponse * response = request->beginResponse(ok ? 200 : 204); request->send(response); } } // namespace emsesp