287 lines
9.1 KiB
C++
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
|