#include #include #include #include "modbus.h" #include "dtu.h" #include "microinverter.h" #include "portParameters.h" Dtu::Dtu(const char *ip_address, int port) { class modbus modbus { ip_address, (uint16_t)port }; this->modbus = std::make_shared(modbus); if (!this->modbus.get()->modbus_connect()) { std::cerr << "NOT CONNECTED" << std::endl; } if (this->modbus.get()->is_connected()) { this->populateMicroinverters(); } } bool Dtu::isConnected() { return this->modbus.get()->is_connected(); } bool Dtu::modbusError() { return this->modbus.get()->err; } std::string Dtu::modbusErrorMessage() { return this->modbus.get()->error_msg; } Dtu::~Dtu() { this->modbus.get()->modbus_close(); } void Dtu::populateMicroinverters() { int portStartAddress = 0x4000; uint16_t registers[19]; int registerCount; registerCount = this->modbus.get()->modbus_read_holding_registers(portStartAddress, 19, registers); if (registerCount != 0) { return; } while (registerCount == 0) { if(registers[0] != 12) { break; } Port port{portStartAddress}; port.setParametersFromMicroinverterArray(registers, 0); 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); } this->getMicroinverterBySerialNumber(port.getParameterByName("microinverterSerialNumber").first.get()->getValue().first.i).first->ports.push_back(port); portStartAddress += 0x0019; registerCount = this->modbus.get()->modbus_read_holding_registers(portStartAddress, 19, registers); } } std::pair Dtu::getMicroinverterBySerialNumber(long long serialNumber) { std::vector::iterator microinvertersIterator = this->microinverters.begin(); while (microinvertersIterator != this->microinverters.end()) { if (microinvertersIterator->serialNumber == serialNumber) { return std::pair(&*microinvertersIterator, true); } else { microinvertersIterator++; } } return std::pair(&*microinvertersIterator, false); } void Dtu::updateMicroinverters(std::vector ¶metersToGet, bool allParameters, std::vector µinvertersToGet) { if (microinvertersToGet.empty()) { std::vector::iterator microinvertersIterator = this->microinverters.begin(); while (microinvertersIterator != this->microinverters.end()) { microinvertersToGet.push_back(microinvertersIterator->serialNumber); microinvertersIterator++; } } std::vector::iterator microinvertersToGetIterator = microinvertersToGet.begin(); while (microinvertersToGetIterator != microinvertersToGet.end()) { std::pair microinverterPair = this->getMicroinverterBySerialNumber(*microinvertersToGetIterator); if (microinverterPair.second) { microinverterPair.first->updateParameters(parametersToGet, allParameters); microinverterPair.first->updateStatusParameters(); } microinvertersToGetIterator++; } } void Dtu::printMicroinverters(std::vector ¶metersToGet, bool allParameters, std::vector µinvertersToGet, bool shortNames, bool printTodayProduction, bool printTotalProduction) { if (microinvertersToGet.empty()) { std::vector::iterator microinvertersIterator = this->microinverters.begin(); while (microinvertersIterator != this->microinverters.end()) { microinvertersToGet.push_back(microinvertersIterator->serialNumber); microinvertersIterator++; } } std::vector::iterator microinvertersToGetIterator = microinvertersToGet.begin(); while (microinvertersToGetIterator != microinvertersToGet.end()) { std::pair microinverterPair = this->getMicroinverterBySerialNumber(*microinvertersToGetIterator); if (microinverterPair.second) { std::cout << " " << "Microinverter: " << microinverterPair.first->serialNumber << std::endl; std::cout << " " << "Microinverter Data Age: " << microinverterPair.first->age << std::endl; if (printTodayProduction) { std::cout << " " << "TodayProduction: " << microinverterPair.first->getTodayProduction() << "Wh" << std::endl; } if (printTotalProduction) { std::cout << " " << "TotalProduction: " << microinverterPair.first->getTotalProduction() << "Wh" << std::endl; } microinverterPair.first->printPorts(parametersToGet, allParameters, shortNames); std::cout << std::endl; } microinvertersToGetIterator++; } } void Dtu::setStatusMicroinverters(uint16_t value, std::string statusName, std::vector& microinvertersToSet) { if (microinvertersToSet.empty()) { std::vector::iterator microinvertersIterator = this->microinverters.begin(); while (microinvertersIterator != this->microinverters.end()) { microinvertersToSet.push_back(microinvertersIterator->serialNumber); microinvertersIterator++; } } std::vector::iterator microinvertersToSetIterator = microinvertersToSet.begin(); while(microinvertersToSetIterator != microinvertersToSet.end()) { std::pair microinverterPair = this->getMicroinverterBySerialNumber(*microinvertersToSetIterator); if(microinverterPair.second) { microinverterPair.first->setStatusWholeMicroinverter(value, statusName); } microinvertersToSetIterator++; } }