etm-powersync-plugins/everest/jsonrpc/everestjsonrpcclient.h
Simon Stürz 648f30c4ab EVerest: Improve connection initialization
The initialization has now a retry logic if there are no data vailable yet.
Retry up to 4 times to create a localhost instance.
The max charging current will be ignored if a 0 value has been received.
2025-10-02 13:29:53 +02:00

293 lines
9.6 KiB
C++

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
*
* Copyright 2013 - 2025, nymea GmbH
* Contact: contact@nymea.io
*
* This file is part of nymea.
* This project including source code and documentation is protected by
* copyright law, and remains the property of nymea GmbH. All rights, including
* reproduction, publication, editing and translation, are reserved. The use of
* this project is subject to the terms of a license agreement to be concluded
* with nymea GmbH in accordance with the terms of use of nymea GmbH, available
* under https://nymea.io/license
*
* GNU Lesser General Public License Usage
* Alternatively, this project may be redistributed and/or modified under the
* terms of the GNU Lesser General Public License as published by the Free
* Software Foundation; version 3. This project 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this project. If not, see <https://www.gnu.org/licenses/>.
*
* For any further details and any questions please contact us under
* contact@nymea.io or see our FAQ/Licensing Information on
* https://nymea.io/license/faq
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
#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