// SPDX-License-Identifier: GPL-2.0-or-later /* * Copyright (C) 2022-2024 Thomas Basler and others */ #include "WebApi_limit.h" #include "WebApi.h" #include "WebApi_errors.h" #include "defaults.h" #include "helper.h" #include #include void WebApiLimitClass::init(AsyncWebServer& server, Scheduler& scheduler) { using std::placeholders::_1; server.on("/api/limit/status", HTTP_GET, std::bind(&WebApiLimitClass::onLimitStatus, this, _1)); server.on("/api/limit/config", HTTP_POST, std::bind(&WebApiLimitClass::onLimitPost, this, _1)); } void WebApiLimitClass::onLimitStatus(AsyncWebServerRequest* request) { if (!WebApi.checkCredentialsReadonly(request)) { return; } AsyncJsonResponse* response = new AsyncJsonResponse(); auto& root = response->getRoot(); for (uint8_t i = 0; i < Hoymiles.getNumInverters(); i++) { auto inv = Hoymiles.getInverterByPos(i); String serial = inv->serialString(); root[serial]["limit_relative"] = inv->SystemConfigPara()->getLimitPercent(); root[serial]["max_power"] = inv->DevInfo()->getMaxPower(); LastCommandSuccess status = inv->SystemConfigPara()->getLastLimitCommandSuccess(); String limitStatus = "Unknown"; if (status == LastCommandSuccess::CMD_OK) { limitStatus = "Ok"; } else if (status == LastCommandSuccess::CMD_NOK) { limitStatus = "Failure"; } else if (status == LastCommandSuccess::CMD_PENDING) { limitStatus = "Pending"; } root[serial]["limit_set_status"] = limitStatus; } WebApi.sendJsonResponse(request, response, __FUNCTION__, __LINE__); } void WebApiLimitClass::onLimitPost(AsyncWebServerRequest* request) { if (!WebApi.checkCredentials(request)) { return; } AsyncJsonResponse* response = new AsyncJsonResponse(); JsonDocument root; if (!WebApi.parseRequestData(request, response, root)) { return; } auto& retMsg = response->getRoot(); if (!(root["serial"].is() && root["limit_value"].is() && root["limit_type"].is())) { retMsg["message"] = "Values are missing!"; retMsg["code"] = WebApiError::GenericValueMissing; WebApi.sendJsonResponse(request, response, __FUNCTION__, __LINE__); return; } // Interpret the string as a hex value and convert it to uint64_t const uint64_t serial = strtoll(root["serial"].as().c_str(), NULL, 16); if (serial == 0) { retMsg["message"] = "Serial must be a number > 0!"; retMsg["code"] = WebApiError::LimitSerialZero; WebApi.sendJsonResponse(request, response, __FUNCTION__, __LINE__); return; } if (root["limit_value"].as() > MAX_INVERTER_LIMIT) { retMsg["message"] = "Limit must between 0 and " STR(MAX_INVERTER_LIMIT) "!"; retMsg["code"] = WebApiError::LimitInvalidLimit; retMsg["param"]["max"] = MAX_INVERTER_LIMIT; WebApi.sendJsonResponse(request, response, __FUNCTION__, __LINE__); return; } if (!((root["limit_type"].as() == PowerLimitControlType::AbsolutNonPersistent) || (root["limit_type"].as() == PowerLimitControlType::AbsolutPersistent) || (root["limit_type"].as() == PowerLimitControlType::RelativNonPersistent) || (root["limit_type"].as() == PowerLimitControlType::RelativPersistent))) { retMsg["message"] = "Invalid type specified!"; retMsg["code"] = WebApiError::LimitInvalidType; WebApi.sendJsonResponse(request, response, __FUNCTION__, __LINE__); return; } float limit = root["limit_value"].as(); PowerLimitControlType type = root["limit_type"].as(); auto inv = Hoymiles.getInverterBySerial(serial); if (inv == nullptr) { retMsg["message"] = "Invalid inverter specified!"; retMsg["code"] = WebApiError::LimitInvalidInverter; WebApi.sendJsonResponse(request, response, __FUNCTION__, __LINE__); return; } inv->sendActivePowerControlRequest(limit, type); retMsg["type"] = "success"; retMsg["message"] = "Settings saved!"; retMsg["code"] = WebApiError::GenericSuccess; WebApi.sendJsonResponse(request, response, __FUNCTION__, __LINE__); }