162 lines
5.3 KiB
C++
162 lines
5.3 KiB
C++
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* Copyright (C) 2022 Thomas Basler and others
|
|
*/
|
|
#include "WebApi_ws_vedirect_live.h"
|
|
#include "AsyncJson.h"
|
|
#include "Configuration.h"
|
|
#include "MessageOutput.h"
|
|
#include "WebApi.h"
|
|
#include "defaults.h"
|
|
#include "PowerLimiter.h"
|
|
|
|
WebApiWsVedirectLiveClass::WebApiWsVedirectLiveClass()
|
|
: _ws("/vedirectlivedata")
|
|
{
|
|
}
|
|
|
|
void WebApiWsVedirectLiveClass::init(AsyncWebServer* server)
|
|
{
|
|
using std::placeholders::_1;
|
|
using std::placeholders::_2;
|
|
using std::placeholders::_3;
|
|
using std::placeholders::_4;
|
|
using std::placeholders::_5;
|
|
using std::placeholders::_6;
|
|
|
|
_server = server;
|
|
_server->on("/api/vedirectlivedata/status", HTTP_GET, std::bind(&WebApiWsVedirectLiveClass::onLivedataStatus, this, _1));
|
|
|
|
_server->addHandler(&_ws);
|
|
_ws.onEvent(std::bind(&WebApiWsVedirectLiveClass::onWebsocketEvent, this, _1, _2, _3, _4, _5, _6));
|
|
}
|
|
|
|
void WebApiWsVedirectLiveClass::loop()
|
|
{
|
|
// see: https://github.com/me-no-dev/ESPAsyncWebServer#limiting-the-number-of-web-socket-clients
|
|
if (millis() - _lastWsCleanup > 1000) {
|
|
_ws.cleanupClients();
|
|
_lastWsCleanup = millis();
|
|
}
|
|
|
|
// do nothing if no WS client is connected
|
|
if (_ws.count() == 0) {
|
|
return;
|
|
}
|
|
|
|
if (millis() - _lastVedirectUpdateCheck < 1000) {
|
|
return;
|
|
}
|
|
_lastVedirectUpdateCheck = millis();
|
|
|
|
uint32_t maxTimeStamp = 0;
|
|
if (VeDirect.getLastUpdate() > maxTimeStamp) {
|
|
maxTimeStamp = VeDirect.getLastUpdate();
|
|
}
|
|
|
|
// Update on ve.direct change or at least after 10 seconds
|
|
if (millis() - _lastWsPublish > (10 * 1000) || (maxTimeStamp != _newestVedirectTimestamp)) {
|
|
|
|
try {
|
|
DynamicJsonDocument root(1024);
|
|
JsonVariant var = root;
|
|
generateJsonResponse(var);
|
|
|
|
String buffer;
|
|
if (buffer) {
|
|
serializeJson(root, buffer);
|
|
|
|
if (Configuration.get().Security_AllowReadonly) {
|
|
_ws.setAuthentication("", "");
|
|
} else {
|
|
_ws.setAuthentication(AUTH_USERNAME, Configuration.get().Security_Password);
|
|
}
|
|
|
|
_ws.textAll(buffer);
|
|
}
|
|
|
|
} catch (std::bad_alloc& bad_alloc) {
|
|
MessageOutput.printf("Call to /api/vedirectlivedata/status temporarely out of resources. Reason: \"%s\".\r\n", bad_alloc.what());
|
|
}
|
|
|
|
_lastWsPublish = millis();
|
|
}
|
|
}
|
|
|
|
void WebApiWsVedirectLiveClass::generateJsonResponse(JsonVariant& root)
|
|
{
|
|
// device info
|
|
root["data_age"] = (millis() - VeDirect.getLastUpdate() ) / 1000;
|
|
root["age_critical"] = !VeDirect.isDataValid();
|
|
root["PID"] = VeDirect.getPidAsString(VeDirect.veFrame.PID);
|
|
root["SER"] = VeDirect.veFrame.SER;
|
|
root["FW"] = VeDirect.veFrame.FW;
|
|
root["LOAD"] = VeDirect.veFrame.LOAD == true ? "ON" : "OFF";
|
|
root["CS"] = VeDirect.getCsAsString(VeDirect.veFrame.CS);
|
|
root["ERR"] = VeDirect.getErrAsString(VeDirect.veFrame.ERR);
|
|
root["OR"] = VeDirect.getOrAsString(VeDirect.veFrame.OR);
|
|
root["MPPT"] = VeDirect.getMpptAsString(VeDirect.veFrame.MPPT);
|
|
root["HSDS"]["v"] = VeDirect.veFrame.HSDS;
|
|
root["HSDS"]["u"] = "Days";
|
|
|
|
// battery info
|
|
root["V"]["v"] = VeDirect.veFrame.V;
|
|
root["V"]["u"] = "V";
|
|
root["I"]["v"] = VeDirect.veFrame.I;
|
|
root["I"]["u"] = "A";
|
|
|
|
// panel info
|
|
root["VPV"]["v"] = VeDirect.veFrame.VPV;
|
|
root["VPV"]["u"] = "V";
|
|
root["PPV"]["v"] = VeDirect.veFrame.PPV;
|
|
root["PPV"]["u"] = "W";
|
|
root["H19"]["v"] = VeDirect.veFrame.H19;
|
|
root["H19"]["u"] = "kWh";
|
|
root["H20"]["v"] = VeDirect.veFrame.H20;
|
|
root["H20"]["u"] = "kWh";
|
|
root["H21"]["v"] = VeDirect.veFrame.H21;
|
|
root["H21"]["u"] = "W";
|
|
root["H22"]["v"] = VeDirect.veFrame.H22;
|
|
root["H22"]["u"] = "kWh";
|
|
root["H23"]["v"] = VeDirect.veFrame.H23;
|
|
root["H23"]["u"] = "W";
|
|
|
|
// power limiter state
|
|
if (Configuration.get().PowerLimiter_Enabled)
|
|
root["PLSTATE"] = PowerLimiter.getPowerLimiterState();
|
|
else
|
|
root["PLSTATE"] = -1;
|
|
root["PLLIMIT"] = PowerLimiter.getLastRequestedPowewrLimit();
|
|
|
|
if (VeDirect.getLastUpdate() > _newestVedirectTimestamp) {
|
|
_newestVedirectTimestamp = VeDirect.getLastUpdate();
|
|
}
|
|
}
|
|
|
|
void WebApiWsVedirectLiveClass::onWebsocketEvent(AsyncWebSocket* server, AsyncWebSocketClient* client, AwsEventType type, void* arg, uint8_t* data, size_t len)
|
|
{
|
|
if (type == WS_EVT_CONNECT) {
|
|
char str[64];
|
|
snprintf(str, sizeof(str), "Websocket: [%s][%u] connect", server->url(), client->id());
|
|
Serial.println(str);
|
|
MessageOutput.println(str);
|
|
} else if (type == WS_EVT_DISCONNECT) {
|
|
char str[64];
|
|
snprintf(str, sizeof(str), "Websocket: [%s][%u] disconnect", server->url(), client->id());
|
|
Serial.println(str);
|
|
MessageOutput.println(str);
|
|
}
|
|
}
|
|
|
|
void WebApiWsVedirectLiveClass::onLivedataStatus(AsyncWebServerRequest* request)
|
|
{
|
|
if (!WebApi.checkCredentialsReadonly(request)) {
|
|
return;
|
|
}
|
|
AsyncJsonResponse* response = new AsyncJsonResponse(false, 1024U);
|
|
JsonVariant root = response->getRoot().as<JsonVariant>();
|
|
generateJsonResponse(root);
|
|
|
|
response->setLength();
|
|
request->send(response);
|
|
} |