powersync-plugins/everest/jsonrpc/everestjsonrpcclient.h

287 lines
9.1 KiB
C++

// SPDX-License-Identifier: GPL-3.0-or-later
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
*
* Copyright (C) 2013 - 2024, nymea GmbH
* Copyright (C) 2024 - 2025, chargebyte austria GmbH
*
* This file is part of nymea-plugins.
*
* nymea-plugins 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.
*
* nymea-plugins 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 nymea-plugins. If not, see <https://www.gnu.org/licenses/>.
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
#ifndef EVERESTJSONRPCCLIENT_H
#define EVERESTJSONRPCCLIENT_H
#include <QObject>
#include <integrations/thing.h>
#include <network/macaddress.h>
#include <network/networkdevicemonitor.h>
#include "everestjsonrpcreply.h"
#include "everestjsonrpcinterface.h"
class EverestJsonRpcClient : public QObject
{
Q_OBJECT
public:
// API Enums
enum ResponseError {
ResponseErrorNoError = 0,
ResponseErrorErrorInvalidParameter,
ResponseErrorErrorOutOfRange,
ResponseErrorErrorValuesNotApplied,
ResponseErrorErrorInvalidEVSEIndex,
ResponseErrorErrorInvalidConnectorId,
ResponseErrorErrorNoDataAvailable,
ResponseErrorErrorUnknownError
};
Q_ENUM(ResponseError)
enum ConnectorType {
ConnectorTypecCCS1,
ConnectorTypecCCS2,
ConnectorTypecG105,
ConnectorTypecTesla,
ConnectorTypecType1,
ConnectorTypecType2,
ConnectorTypes309_1P_16A,
ConnectorTypes309_1P_32A,
ConnectorTypes309_3P_16A,
ConnectorTypes309_3P_32A,
ConnectorTypesBS1361,
ConnectorTypesCEE_7_7,
ConnectorTypesType2,
ConnectorTypesType3,
ConnectorTypeOther1PhMax16A,
ConnectorTypeOther1PhOver16A,
ConnectorTypeOther3Ph,
ConnectorTypePan,
ConnectorTypewInductive,
ConnectorTypewResonant,
ConnectorTypeUndetermined
};
Q_ENUM(ConnectorType)
enum ChargeProtocol {
ChargeProtocolUnknown,
ChargeProtocolIEC61851,
ChargeProtocolDIN70121,
ChargeProtocolISO15118,
ChargeProtocolISO15118_20
};
Q_ENUM(ChargeProtocol)
enum EvseState {
EvseStateUnplugged,
EvseStateDisabled,
EvseStatePreparing,
EvseStateReserved,
EvseStateAuthRequired,
EvseStateWaitingForEnergy,
EvseStateCharging,
EvseStateChargingPausedEV,
EvseStateChargingPausedEVSE,
EvseStateFinished,
EvseStateSwitchingPhases
};
Q_ENUM(EvseState)
// API Objects
typedef struct ChargerInfo {
QString vendor;
QString model;
QString serialNumber;
QString firmwareVersion;
} ChargerInfo;
typedef struct ConnectorInfo {
int connectorId = -1;
ConnectorType type = ConnectorTypeUndetermined;
QString description; // optional
} ConnectorInfo;
typedef struct EVSEInfo {
int index = -1;
QString id;
QString description; // optional
bool bidirectionalCharging = false;
QList<ConnectorInfo> availableConnectors;
} EVSEInfo;
typedef struct ACChargeStatus {
int activePhaseCount = 3;
} ACChargeStatus;
typedef struct ACChargeParameters {
double maxCurrent = 0; // A
double maxPhaseCount = 0;
// Not supported yet with 1.0.0
// double maxChargePower = 0; // W
// double minChargePower = 0; // W
// double nominalFrequency = 0; // Hz
} ACChargeParameters;
typedef struct EVSEStatus {
double chargedEnergyWh = 0;
double dischargedEnergyWh = 0;
int chargingDuration = 0; // seconds
bool chargingAllowed = false;
bool available = false;
int activeConnectorIndex = -1;
bool errorPresent = false;
ChargeProtocol chargeProtocol = ChargeProtocolUnknown;
EvseState evseState = EvseStateUnplugged;
QString evseStateString;
ACChargeStatus acChargeStatus; // optional
ACChargeParameters acChargeParameters; // optional
// TODO:
// o: "dc_charge_param": "$DCChargeParametersObj",
// o: "dc_charge_status": "$DCChargeLoopObj",
// o: display_parameters: "$DisplayParametersObj",
} EVSEStatus;
typedef struct HardwareCapabilities {
double maxCurrentExport = 0;
double maxCurrentImport = 0;
int maxPhaseCountExport = 0;
int maxPhaseCountImport = 0;
double minCurrentExport = 0;
double minCurrentImport = 0;
int minPhaseCountExport = 0;
int minPhaseCountImport = 0;
bool phaseSwitchDuringCharging = false;
} HardwareCapabilities;
typedef struct MeterData {
QString meterId;
QString serialNumber;
bool phaseSequenceError = false;
//quint64 timestamp = 0;
float powerL1 = 0; // W
float powerL2 = 0; // W
float powerL3 = 0; // W
float powerTotal = 0; // W
float currentL1 = 0; // A
float currentL2 = 0; // A
float currentL3 = 0; // A
float currentN = 0; // A
float voltageL1 = 0; // V
float voltageL2 = 0; // V
float voltageL3 = 0; // V
float energyImportedL1 = 0; // Wh
float energyImportedL2 = 0; // Wh
float energyImportedL3 = 0; // Wh
float energyImportedTotal = 0; // Wh
float energyExportedL1 = 0; // Wh
float energyExportedL2 = 0; // Wh
float energyExportedL3 = 0; // Wh
float energyExportedTotal = 0; // Wh
float frequencyL1 = 0; // Hz
float frequencyL2 = 0; // Hz
float frequencyL3 = 0; // Hz
} MeterData;
explicit EverestJsonRpcClient(QObject *parent = nullptr);
QUrl serverUrl();
bool available() const;
// Once available, following properties are set
QString apiVersion() const;
QString everestVersion() const;
ChargerInfo chargerInfo() const;
QList<EVSEInfo> evseInfos() const;
// API calls
EverestJsonRpcReply *evseGetInfo(int evseIndex);
EverestJsonRpcReply *evseGetStatus(int evseIndex);
EverestJsonRpcReply *evseGetHardwareCapabilities(int evseIndex);
EverestJsonRpcReply *evseGetMeterData(int evseIndex);
EverestJsonRpcReply *evseSetChargingAllowed(int evseIndex, bool allowed);
EverestJsonRpcReply *evseSetACChargingCurrent(int evseIndex, double current);
EverestJsonRpcReply *evseSetACChargingPhaseCount(int evseIndex, int phaseCount);
// API parser methods
// Enums
static ResponseError parseResponseError(const QString &responseErrorString);
static ConnectorType parseConnectorType(const QString &connectorTypeString);
static ChargeProtocol parseChargeProtocol(const QString &chargeProtocolString);
static EvseState parseEvseState(const QString &evseStateString);
// Objects
static EVSEInfo parseEvseInfo(const QVariantMap &evseInfoMap);
static ConnectorInfo parseConnectorInfo(const QVariantMap &connectorInfoMap);
static EVSEStatus parseEvseStatus(const QVariantMap &evseStatusMap);
static ACChargeStatus parseACChargeStatus(const QVariantMap &acChargeStatusMap);
static ACChargeParameters parseACChargeParameters(const QVariantMap &acChargeParametersMap);
static HardwareCapabilities parseHardwareCapabilities(const QVariantMap &hardwareCapabilitiesMap);
static MeterData parseMeterData(const QVariantMap &meterDataMap);
public slots:
void connectToServer(const QUrl &serverUrl);
void disconnectFromServer();
signals:
void connectionErrorOccurred();
void availableChanged(bool available);
// Notifications
void evseStatusChanged(int evseIndex, const EverestJsonRpcClient::EVSEStatus &evseStatus);
void hardwareCapabilitiesChanged(int evseIndex, const EverestJsonRpcClient::HardwareCapabilities &hardwareCapabilities);
void meterDataChanged(int evseIndex, const EverestJsonRpcClient::MeterData &hardwareCapabilities);
// void activeErrorsChanged(); // TODO
private slots:
void sendRequest(EverestJsonRpcReply *reply);
void processDataPacket(const QByteArray &data);
private:
bool m_available = false;
int m_commandId = 0;
EverestJsonRpcInterface *m_interface = nullptr;
QHash<int, EverestJsonRpcReply *> m_replies;
EverestJsonRpcReply *createReply(QString method, QVariantMap params = QVariantMap(), bool retry = false);
// Init infos
QString m_apiVersion;
QString m_everestVersion;
ChargerInfo m_chargerInfo;
bool m_authenticationRequired = false;
QList<EVSEInfo> m_evseInfos;
int getNextCommandId();
// API calls
EverestJsonRpcReply *apiHello();
EverestJsonRpcReply *chargePointGetEVSEInfos();
};
#endif // EVERESTJSONRPCCLIENT_H