hoymilesClient/src/hoymiles/dtu.cpp

246 lines
8.7 KiB
C++
Raw Normal View History

2024-04-09 20:31:43 +02:00
#include <chrono>
2024-03-16 21:15:15 +01:00
#include <iostream>
2024-03-20 11:28:43 +01:00
#include <string>
2024-04-09 12:21:21 +02:00
#include <thread>
2024-04-09 20:31:43 +02:00
#include <vector>
2024-03-16 21:15:15 +01:00
#include "modbus.h"
#include "dtu.h"
#include "microinverter.h"
#include "portParameters.h"
2024-04-18 18:27:57 +02:00
Dtu::Dtu(const char *address, int id, bool rtu, bool tcp) : dtuPort(0) {
dtuPort.statusPortStartAddress = 0xd000;
2024-04-09 20:31:43 +02:00
if (tcp) {
2024-04-09 12:21:21 +02:00
this->modbus = modbus_new_tcp(address, id);
}
2024-04-09 20:31:43 +02:00
if (rtu) {
2024-04-09 12:21:21 +02:00
this->modbus = modbus_new_rtu(address, 9600, 'N', 8, 1);
modbus_rtu_set_serial_mode(this->modbus, MODBUS_RTU_RS485);
}
2024-04-09 12:21:21 +02:00
this->connected = false;
if (modbus_connect(this->modbus) == -1) {
2024-04-18 18:27:57 +02:00
std::cerr << "[" << id << "] NOT_CONNECTED" << std::endl;
2024-04-09 20:31:43 +02:00
} else {
2024-04-09 12:21:21 +02:00
this->connected = true;
2024-04-09 20:31:43 +02:00
if (rtu) {
2024-04-09 12:21:21 +02:00
modbus_set_slave(this->modbus, id);
}
this->populateMicroinverters();
}
}
2024-04-18 18:27:57 +02:00
Dtu::Dtu(modbus_t *modbus, int id = -1) : dtuPort(0) {
dtuPort.statusPortStartAddress = 0xd000;
this->modbus = modbus;
this->rtuId = id;
this->connected = false;
if (modbus_connect(this->modbus) == -1) {
std::cerr << "[" << id << "] NOT_CONNECTED" << std::endl;
} else {
this->connected = true;
if (id != -1) {
modbus_set_slave(this->modbus, rtuId);
}
this->populateMicroinverters();
}
}
2024-04-09 12:21:21 +02:00
bool Dtu::isConnected() { return this->connected; }
2024-03-16 21:15:15 +01:00
2024-04-09 12:21:21 +02:00
Dtu::~Dtu() {
modbus_close(this->modbus);
modbus_free(this->modbus);
}
2024-03-16 21:15:15 +01:00
void Dtu::populateMicroinverters() {
2024-04-18 18:27:57 +02:00
if(this->rtuId != -1) {
modbus_set_slave(this->modbus, this->rtuId);
}
int portStartAddress = 0x4000;
2024-04-18 18:27:57 +02:00
int registersToRead{19};
uint16_t registers[registersToRead];
bool lastSuccesful{true};
int addressToSkip{-1};
2024-04-09 20:47:44 +02:00
while (portStartAddress <= (0x4000 + (0x0019 * 99))) {
2024-04-18 18:27:57 +02:00
int registerCount{-1};
int timesTried{0};
while (((timesTried < 3) && (lastSuccesful && registerCount == -1) && portStartAddress != addressToSkip)) {
registerCount = modbus_read_registers(this->modbus, portStartAddress, registersToRead, registers);
timesTried++;
}
2024-04-09 20:47:44 +02:00
portStartAddress += 0x0019;
2024-04-18 18:27:57 +02:00
if ((registers[0] == 12 && registerCount != -1) && portStartAddress != addressToSkip) {
if (!lastSuccesful) {
addressToSkip = portStartAddress;
portStartAddress -= 2 * 0x0019;
}
lastSuccesful = true;
2024-04-09 20:31:43 +02:00
Port port{portStartAddress};
port.setParametersFromMicroinverterArray(registers, 0);
2024-04-09 20:31:43 +02:00
if (!this->getMicroinverterBySerialNumber(port.getParameterByName("microinverterSerialNumber").first.get()->getValue().first.i).second) {
Microinverter microinverter{this->modbus, portStartAddress, port.getParameterByName("microinverterSerialNumber").first.get()->getValue().first.i};
this->microinverters.push_back(microinverter);
}
2024-04-18 18:27:57 +02:00
Microinverter *microinverter = this->getMicroinverterBySerialNumber(port.getParameterByName("microinverterSerialNumber").first.get()->getValue().first.i).first;
std::vector<Port>::iterator portsIterator = microinverter->ports.begin();
bool valueExists{false};
while (portsIterator != microinverter->ports.end() && !valueExists) {
if (portsIterator->getParameterByName("portNumber").first.get()->getValue().first.i == port.getParameterByName("portNumber").first.get()->getValue().first.i) {
valueExists = true;
}
portsIterator++;
}
if (!valueExists) {
microinverter->ports.push_back(port);
}
} else {
lastSuccesful = false;
}
2024-04-18 18:27:57 +02:00
std::this_thread::sleep_for(std::chrono::milliseconds(20));
}
}
std::pair<Microinverter *, bool> Dtu::getMicroinverterBySerialNumber(long long serialNumber) {
std::vector<Microinverter>::iterator microinvertersIterator = this->microinverters.begin();
2024-03-20 19:55:36 +01:00
while (microinvertersIterator != this->microinverters.end()) {
if (microinvertersIterator->serialNumber == serialNumber) {
return std::pair<Microinverter *, bool>(&*microinvertersIterator, true);
2024-03-20 19:55:36 +01:00
} else {
microinvertersIterator++;
}
}
return std::pair<Microinverter *, bool>(&*microinvertersIterator, false);
2024-03-16 21:15:15 +01:00
}
2024-03-20 16:39:25 +01:00
void Dtu::updateMicroinverters(std::vector<std::string> &parametersToGet, bool allParameters, std::vector<long long> &microinvertersToGet) {
2024-04-18 18:27:57 +02:00
if(this->rtuId != -1) {
modbus_set_slave(this->modbus, this->rtuId);
}
2024-03-20 19:55:36 +01:00
if (microinvertersToGet.empty()) {
std::vector<Microinverter>::iterator microinvertersIterator = this->microinverters.begin();
2024-03-20 19:55:36 +01:00
while (microinvertersIterator != this->microinverters.end()) {
microinvertersToGet.push_back(microinvertersIterator->serialNumber);
microinvertersIterator++;
}
2024-03-16 21:15:15 +01:00
}
2024-03-20 16:39:25 +01:00
std::vector<long long>::iterator microinvertersToGetIterator = microinvertersToGet.begin();
2024-03-20 19:55:36 +01:00
while (microinvertersToGetIterator != microinvertersToGet.end()) {
std::pair<Microinverter *, bool> microinverterPair = this->getMicroinverterBySerialNumber(*microinvertersToGetIterator);
if (microinverterPair.second) {
microinverterPair.first->updateParameters(parametersToGet, allParameters);
2024-04-06 00:32:49 +02:00
microinverterPair.first->updateStatusParameters();
}
microinvertersToGetIterator++;
2024-03-20 11:28:43 +01:00
}
}
void Dtu::printMicroinverters(std::vector<std::string> &parametersToGet, bool allParameters, std::vector<long long> &microinvertersToGet, bool shortNames, bool printTodayProduction, bool printTotalProduction) {
2024-03-20 19:55:36 +01:00
if (microinvertersToGet.empty()) {
std::vector<Microinverter>::iterator microinvertersIterator = this->microinverters.begin();
2024-03-20 19:55:36 +01:00
while (microinvertersIterator != this->microinverters.end()) {
microinvertersToGet.push_back(microinvertersIterator->serialNumber);
microinvertersIterator++;
}
}
2024-03-20 11:28:43 +01:00
2024-03-20 16:39:25 +01:00
std::vector<long long>::iterator microinvertersToGetIterator = microinvertersToGet.begin();
2024-03-20 19:55:36 +01:00
while (microinvertersToGetIterator != microinvertersToGet.end()) {
std::pair<Microinverter *, bool> microinverterPair = this->getMicroinverterBySerialNumber(*microinvertersToGetIterator);
if (microinverterPair.second) {
2024-04-09 20:31:43 +02:00
std::cout << " "
<< "Microinverter: " << microinverterPair.first->serialNumber << std::endl;
std::cout << " "
<< "Microinverter Data Age: " << microinverterPair.first->age << std::endl;
if (printTodayProduction) {
2024-04-09 20:31:43 +02:00
std::cout << " "
<< "TodayProduction: " << microinverterPair.first->getTodayProduction() << "Wh" << std::endl;
}
if (printTotalProduction) {
2024-04-09 20:31:43 +02:00
std::cout << " "
<< "TotalProduction: " << microinverterPair.first->getTotalProduction() << "Wh" << std::endl;
}
microinverterPair.first->printPorts(parametersToGet, allParameters, shortNames);
std::cout << std::endl;
}
microinvertersToGetIterator++;
2024-03-20 11:28:43 +01:00
}
2024-04-06 16:29:02 +02:00
}
2024-04-09 20:31:43 +02:00
void Dtu::setStatusMicroinverters(uint16_t value, std::string statusName, std::vector<long long> &microinvertersToSet) {
2024-04-06 16:29:02 +02:00
if (microinvertersToSet.empty()) {
std::vector<Microinverter>::iterator microinvertersIterator = this->microinverters.begin();
while (microinvertersIterator != this->microinverters.end()) {
microinvertersToSet.push_back(microinvertersIterator->serialNumber);
microinvertersIterator++;
}
}
std::vector<long long>::iterator microinvertersToSetIterator = microinvertersToSet.begin();
2024-04-09 20:31:43 +02:00
while (microinvertersToSetIterator != microinvertersToSet.end()) {
2024-04-06 16:29:02 +02:00
std::pair<Microinverter *, bool> microinverterPair = this->getMicroinverterBySerialNumber(*microinvertersToSetIterator);
2024-04-09 20:31:43 +02:00
if (microinverterPair.second) {
2024-04-06 16:29:02 +02:00
microinverterPair.first->setStatusWholeMicroinverter(value, statusName);
}
microinvertersToSetIterator++;
}
2024-04-06 19:29:23 +02:00
}
2024-04-09 20:31:43 +02:00
bool Dtu::empty() { return this->microinverters.empty(); }
2024-04-09 12:21:21 +02:00
void Dtu::listOfMicroinverters() {
std::vector<Microinverter>::iterator microinvertersIterator = this->microinverters.begin();
std::cout << "Microinverter list:" << std::endl;
2024-04-09 20:31:43 +02:00
while (microinvertersIterator != this->microinverters.end()) {
2024-04-09 12:21:21 +02:00
std::cout << " " << microinvertersIterator->serialNumber << std::endl;
microinvertersIterator++;
}
2024-04-18 18:27:57 +02:00
}
float Dtu::getCurrentPower() {
std::vector<Microinverter>::iterator microinvertersIterator = this->microinverters.begin();
float currentPower{0};
while (microinvertersIterator != this->microinverters.end()) {
currentPower += microinvertersIterator->getCurrentPower();
microinvertersIterator++;
}
return currentPower;
}
int Dtu::getCurrentOnOff() {
if(this->microinverters.size() == 0) {
return 0;
}
std::vector<Microinverter>::iterator microinvertersIterator = this->microinverters.begin();
int currentOn{0};
while(microinvertersIterator != this->microinverters.end()) {
currentOn += microinvertersIterator->getCurrentOnOff();
microinvertersIterator++;
}
float balance = currentOn / this->microinverters.size();
if(balance >= 0.5) {
return 1;
}
else {
return 0;
}
}
void Dtu::turnOffMicroinverters() { this->dtuPort.getStatusByName("onOff").first.get()->writeValue(0, this->modbus, this->dtuPort.statusPortStartAddress); }
void Dtu::turnOnMicroinverters() { this->dtuPort.getStatusByName("onOff").first.get()->writeValue(1, this->modbus, this->dtuPort.statusPortStartAddress); }
void Dtu::limitMicroinverters(uint16_t limit) { this->dtuPort.getStatusByName("limitActivePower").first.get()->writeValue(limit, this->modbus, this->dtuPort.statusPortStartAddress); }