diff --git a/CODEOWNERS b/CODEOWNERS index 59a56ddd9f1d3..d8f726f03add5 100644 --- a/CODEOWNERS +++ b/CODEOWNERS @@ -216,6 +216,7 @@ /bundles/org.openhab.binding.modbus.sbc/ @fwolter /bundles/org.openhab.binding.modbus.stiebeleltron/ @pail23 /bundles/org.openhab.binding.modbus.studer/ @giovannimirulla +/bundles/org.openhab.binding.modbus.sungrow/ @soenkekueper /bundles/org.openhab.binding.modbus.sunspec/ @mrbig /bundles/org.openhab.binding.monopriceaudio/ @mlobstein /bundles/org.openhab.binding.mpd/ @stefanroellin diff --git a/bom/openhab-addons/pom.xml b/bom/openhab-addons/pom.xml index dd5cb09900a0a..520625b1380bb 100644 --- a/bom/openhab-addons/pom.xml +++ b/bom/openhab-addons/pom.xml @@ -1071,6 +1071,11 @@ org.openhab.binding.modbus.studer ${project.version} + + org.openhab.addons.bundles + org.openhab.binding.modbus.sungrow + ${project.version} + org.openhab.addons.bundles org.openhab.binding.modbus.sunspec diff --git a/bundles/org.openhab.binding.modbus.sungrow/NOTICE b/bundles/org.openhab.binding.modbus.sungrow/NOTICE new file mode 100644 index 0000000000000..38d625e349232 --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/NOTICE @@ -0,0 +1,13 @@ +This content is produced and maintained by the openHAB project. + +* Project home: https://www.openhab.org + +== Declared Project Licenses + +This program and the accompanying materials are made available under the terms +of the Eclipse Public License 2.0 which is available at +https://www.eclipse.org/legal/epl-2.0/. + +== Source Code + +https://github.com/openhab/openhab-addons diff --git a/bundles/org.openhab.binding.modbus.sungrow/README.md b/bundles/org.openhab.binding.modbus.sungrow/README.md new file mode 100644 index 0000000000000..79a87f2cf34b6 --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/README.md @@ -0,0 +1,174 @@ +# Modbus Sungrow Binding + +This binding integrates the sungrow inverters into openHAB. +It is based on the Sungrow specification "Communication Protocol of Residential Hybrid Inverter V1.0.23", which can be found here: https://github.com/bohdan-s/SunGather/issues/36. + +## Supported Inverters + +As defined within the spec mentioned above the following inverters are supported, but not all are tested yet: + +- SH3K6 +- SH4K6 +- SH5K-20 +- SH5K-V13 +- SH3K6-30 +- SH4K6-30 +- SH5K-30 +- SH3.0RS +- SH3.6RS +- SH4.0RS +- SH5.0RS +- SH6.0RS +- SH5.0RT +- SH6.0RT +- SH8.0RT +- SH10RT + +## Supported Things + +The binding supports only one thing: + +- `sungrow-inverter`: The sungrow inverter + +## Preparation + +The data from the inverter is read via Modbus. So you need to configure a Modbus Serial Slave `serial` or Modbus TCP Slave `tcp` as bridge first. +If you are using a Modbus TCP Slave and the WiNet-S Communication Module please ensure: + +- that you have the correct IP-Address of your WiNet-S Device +- that Modbus is enabled within the Communication Module +- that you've the correct port number +- that the white list is disabled or your openHAB instance IP is listed + +Enabling modbus and whitelist setting can be done in WiNet-S Web-UI as shown below: +WiNet-S Modbus configuration + +## Thing Configuration + +Once you've configured the Modbus TCP Slave or Modbus Serial Slave as Bridge you can configure the Sungrow inverter thing. +You just have to select the configured bridge and optional configure the polling interval. + +### Sungrow Inverter (`sungrow-inverter`) + +| Name | Type | Description | Default | Required | Advanced | +|--------------|---------|--------------------------------------|---------|----------|----------| +| pollInterval | integer | Interval the device is polled in ms. | 5000 | yes | no | + +## Channels + +The `sungrow-inverter` thing has channels that serve the current state of the sungrow inverter, as you are used to from the iSolareCloud Website and App. + +| Channel Type ID | Item Type | Description | Advanced | Channel Group | +|------------------------------------|--------------------------|---------------------------------------|-----------|---------------------| +| sg-internal-temperature | Number:Temperature | Internal Temperature | yes | Overview | +| sg-total-dc-power | Number:Power | Total DC Power | no | Overview | +| sg-phase-a-voltage | Number:ElectricPotential | Phase A Voltage | yes | Overview | +| sg-phase-b-voltage | Number:ElectricPotential | Phase B Voltage | yes | Overview | +| sg-phase-c-voltage | Number:ElectricPotential | Phase C Voltage | yes | Overview | +| sg-daily-pv-generation | Number:Energy | Daily PV Generation | no | Overview | +| sg-total-pv-generation | Number:Energy | Total PV Generation | no | Overview | +| sg-reactive-power | Number:Power | Reactive Power | yes | Overview | +| sg-power-factor | Number:Dimensionless | Power Factor | yes | Overview | +| sg-phase-a-current | Number:ElectricCurrent | Phase A Current | yes | Overview | +| sg-phase-b-current | Number:ElectricCurrent | Phase B Current | yes | Overview | +| sg-phase-c-current | Number:ElectricCurrent | Phase C Current | yes | Overview | +| sg-total-active-power | Number:Power | Total Active Power | no | Overview | +| sg-grid-frequency | Number:Frequency | Grid Frequency | yes | Overview | +| sg-mppt1-voltage | Number:ElectricPotential | MPPT1 Voltage | yes | MPPT Information | +| sg-mppt1-current | Number:ElectricCurrent | MPPT1 Current | yes | MPPT Information | +| sg-mppt2-voltage | Number:ElectricPotential | MPPT2 Voltage | yes | MPPT Information | +| sg-mppt2-current | Number:ElectricCurrent | MPPT2 Current | yes | MPPT Information | +| sg-daily-battery-charge | Number:Energy | Daily Battery Charge | no | Battery Information | +| sg-total-battery-charge | Number:Energy | Total Battery Charge | no | Battery Information | +| sg-battery-voltage | Number:ElectricPotential | Battery Voltage | yes | Battery Information | +| sg-battery-current | Number:ElectricCurrent | Battery Current | yes | Battery Information | +| sg-battery-power | Number:Power | Battery Power | no | Battery Information | +| sg-battery-level | Number:Dimensionless | Battery Level | no | Battery Information | +| sg-battery-healthy | Number:Dimensionless | Battery Healthy | no | Battery Information | +| sg-battery-temperature | Number:Temperature | Battery Temperature | no | Battery Information | +| sg-daily-battery-discharge-energy | Number:Energy | Daily Battery Discharge Energy | no | Battery Information | +| sg-total-battery-discharge-energy | Number:Energy | Total Battery Discharge Energy | no | Battery Information | +| sg-battery-capacity | Number:Energy | Battery Capacity | no | Battery Information | +| sg-daily-charge-energy | Number:Energy | Daily Charge Energy | no | Battery Information | +| sg-total-charge-energy | Number:Energy | Total Charge Energy | no | Battery Information | +| sg-daily-import-energy | Number:Energy | Daily Import Energy | no | Grid Information | +| sg-total-import-energy | Number:Energy | Total Import Energy | no | Grid Information | +| sg-daily-export-energy | Number:Energy | Daily Export Energy | no | Grid Information | +| sg-total-export-energy | Number:Energy | Total Export Energy | no | Grid Information | +| sg-daily-export-power-from-pv | Number:Power | Daily Export Power from PV | no | Grid Information | +| sg-total-export-energy-from-pv | Number:Energy | Total Export Energy from PV | no | Grid Information | +| sg-export-power | Number:Power | Export Power | no | Grid Information | +| sg-load-power | Number:Power | Load Power | no | Load Information | +| sg-daily-direct-energy-consumption | Number:Energy | Daily Direct Energy Consumption | no | Load Information | +| sg-total-direct-energy-consumption | Number:Energy | Total Direct Energy Consumption | no | Load Information | +| sg-self-consumption-today | Number:Dimensionless | Self Consumption Today | no | Load Information | + +## Full Example + +This example shows how to configure a sungrow inverter connected via modbus and uses the most common channels. + +_sungrow.things_ + +```java +Bridge modbus:tcp:sungrowBridge [ host="10.0.0.2", port=502, id=1, enableDiscovery=false ] { + Thing sungrow-inverter sungrowInverter "Sungrow Inverter" [ pollInterval=5000 ] +} +``` + +_sungrow.items_ + +```java +// Groups +Group sungrowInverter "Sungrow Inverter" ["Inverter"] +Group overview "Overview" (sungrowInverter) +Group batteryInformation "Battery information" (sungrowInverter) +Group gridInformation "Grid information" (sungrowInverter) +Group loadInformation "Load information" (sungrowInverter) + +// Overview +Number:Power total_active_power "Total Active Power" (overview) ["Measurement", "Power"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-overview#sg-total-active-power"} +Number:Power total_dc_power "Total DC Power" (overview) ["Measurement", "Power"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-overview#sg-total-dc-power"} +Number:Energy daily_pv_generation "Daily PV Generation" (overview) ["Measurement", "Energy"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-overview#sg-daily-pv-generation"} +Number:Energy total_pv_generation "Total PV Generation" (overview) ["Measurement", "Energy"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-overview#sg-total-pv-generation"} + +// Battery information +Number:Power battery_power "Battery Power" (batteryInformation) ["Measurement", "Power"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-battery-information#sg-battery-power"} +Number:Dimensionless battery_level "Battery Level" (batteryInformation) ["Measurement", "Energy"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-battery-information#sg-battery-level"} +Number:Energy daily_charge_energy "Daily Battery Charge Energy" (batteryInformation) ["Measurement", "Energy"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-battery-information#sg-daily-charge-energy"} +Number:Energy daily_discharge_energy "Daily Battery Discharge Energy" (batteryInformation) ["Measurement", "Energy"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-battery-information#sg-daily-battery-discharge-energy"} + +// Grid information +Number:Power export_power "Export Power" (gridInformation) ["Measurement", "Power"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-grid-information#sg-export-power"} +Number:Energy daily_export_energy "Daily Export Energy" (gridInformation) ["Measurement", "Energy"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-grid-information#sg-daily-export-energy"} +Number:Energy daily_import_energy "Daily Import Energy" (gridInformation) ["Measurement", "Energy"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-grid-information#sg-daily-import-energy"} + +// Load information +Number:Power load_power "Load Power" (loadInformation) ["Measurement", "Power"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-load-information#sg-load-power"} +Number:Energy daily_direct_energy_consumption "Daily Direct Energy Consumption" (loadInformation) ["Measurement", "Energy"] {channel="modbus:sungrow-inverter:sungrowBridge:sungrowInverter:sg-load-information#sg-daily-direct-energy-consumption"} +``` + +_sungrow.sitemap_ + +```perl +sitemap sungrow label="Sungrow Binding" +{ + Frame { + Text item=total_active_power + Text item=total_dc_power + Text item=daily_pv_generation + Text item=total_pv_generation + + Text item=battery_power + Text item=battery_level + Text item=daily_charge_energy + Text item=daily_discharge_energy + + Text item=export_power + Text item=daily_export_energy + Text item=daily_import_energy + + Text item=load_power + Text item=daily_direct_energy_consumption + } +} +``` diff --git a/bundles/org.openhab.binding.modbus.sungrow/doc/WiNet-S_Modbus.PNG b/bundles/org.openhab.binding.modbus.sungrow/doc/WiNet-S_Modbus.PNG new file mode 100644 index 0000000000000..942273fdc0877 Binary files /dev/null and b/bundles/org.openhab.binding.modbus.sungrow/doc/WiNet-S_Modbus.PNG differ diff --git a/bundles/org.openhab.binding.modbus.sungrow/pom.xml b/bundles/org.openhab.binding.modbus.sungrow/pom.xml new file mode 100644 index 0000000000000..63dbb502dd4a5 --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/pom.xml @@ -0,0 +1,27 @@ + + + + 4.0.0 + + + org.openhab.addons.bundles + org.openhab.addons.reactor.bundles + 4.2.0-SNAPSHOT + + + org.openhab.binding.modbus.sungrow + + openHAB Add-ons :: Bundles :: Modbus Sungrow Binding + + + + org.openhab.addons.bundles + org.openhab.binding.modbus + ${project.version} + provided + + + + + diff --git a/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/ConversionConstants.java b/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/ConversionConstants.java new file mode 100644 index 0000000000000..9d7298bd317fc --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/ConversionConstants.java @@ -0,0 +1,42 @@ +/** + * Copyright (c) 2010-2024 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.modbus.sungrow.internal; + +import java.math.BigDecimal; +import java.math.BigInteger; +import java.util.function.Function; + +import org.eclipse.jdt.annotation.NonNullByDefault; + +/** + * Constants for converting values. + * + * @author Sönke Küper - Initial contribution + */ +@NonNullByDefault +final class ConversionConstants { + + private ConversionConstants() { + } + + /** + * Multiplicand for 0.1. + */ + static final BigDecimal DIV_BY_TEN = new BigDecimal(BigInteger.ONE, 1); + + /** + * Value conversion from Celsius to Kelvin. + */ + static final Function CELSIUS_TO_KELVIN = (BigDecimal celsius) -> celsius + .add(new BigDecimal(273.15f)); +} diff --git a/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/ModbusSungrowBindingConstants.java b/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/ModbusSungrowBindingConstants.java new file mode 100644 index 0000000000000..1b1abaeacb3e5 --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/ModbusSungrowBindingConstants.java @@ -0,0 +1,33 @@ +/** + * Copyright (c) 2010-2024 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.modbus.sungrow.internal; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.openhab.binding.modbus.ModbusBindingConstants; +import org.openhab.core.thing.ThingTypeUID; + +/** + * The {@link ModbusSungrowBindingConstants} class defines common constants, which are + * used across the whole binding. + * + * @author Sönke Küper - Initial contribution + */ +@NonNullByDefault +public class ModbusSungrowBindingConstants { + + /** + * ThingType-ID for Inverter. + */ + public static final ThingTypeUID THING_TYPE_INVERTER = new ThingTypeUID(ModbusBindingConstants.BINDING_ID, + "sungrow-inverter"); +} diff --git a/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/ModbusSungrowHandlerFactory.java b/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/ModbusSungrowHandlerFactory.java new file mode 100644 index 0000000000000..d0a547cd18022 --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/ModbusSungrowHandlerFactory.java @@ -0,0 +1,55 @@ +/** + * Copyright (c) 2010-2024 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.modbus.sungrow.internal; + +import static org.openhab.binding.modbus.sungrow.internal.ModbusSungrowBindingConstants.*; + +import java.util.Set; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.openhab.core.thing.Thing; +import org.openhab.core.thing.ThingTypeUID; +import org.openhab.core.thing.binding.BaseThingHandlerFactory; +import org.openhab.core.thing.binding.ThingHandler; +import org.openhab.core.thing.binding.ThingHandlerFactory; +import org.osgi.service.component.annotations.Component; + +/** + * The {@link ModbusSungrowHandlerFactory} is responsible for creating things and thing + * handlers. + * + * @author Sönke Küper - Initial contribution + */ +@NonNullByDefault +@Component(configurationPid = "binding.sungrow", service = ThingHandlerFactory.class) +public class ModbusSungrowHandlerFactory extends BaseThingHandlerFactory { + + private static final Set SUPPORTED_THING_TYPES_UIDS = Set.of(THING_TYPE_INVERTER); + + @Override + public boolean supportsThingType(ThingTypeUID thingTypeUID) { + return SUPPORTED_THING_TYPES_UIDS.contains(thingTypeUID); + } + + @Override + protected @Nullable ThingHandler createHandler(Thing thing) { + ThingTypeUID thingTypeUID = thing.getThingTypeUID(); + + if (THING_TYPE_INVERTER.equals(thingTypeUID)) { + return new SungrowInverterHandler(thing); + } + + return null; + } +} diff --git a/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/SungrowInverterConfiguration.java b/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/SungrowInverterConfiguration.java new file mode 100644 index 0000000000000..88965f9c52024 --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/SungrowInverterConfiguration.java @@ -0,0 +1,26 @@ +/** + * Copyright (c) 2010-2024 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.modbus.sungrow.internal; + +import org.eclipse.jdt.annotation.NonNullByDefault; + +/** + * The {@link SungrowInverterConfiguration} class contains fields mapping thing configuration parameters. + * + * @author Sönke Küper - Initial contribution + */ +@NonNullByDefault +public class SungrowInverterConfiguration { + + public int pollInterval; +} diff --git a/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/SungrowInverterHandler.java b/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/SungrowInverterHandler.java new file mode 100644 index 0000000000000..5b1d55f793465 --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/SungrowInverterHandler.java @@ -0,0 +1,185 @@ +/** + * Copyright (c) 2010-2024 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.modbus.sungrow.internal; + +import java.util.ArrayDeque; +import java.util.ArrayList; +import java.util.Deque; +import java.util.List; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.openhab.binding.modbus.handler.BaseModbusThingHandler; +import org.openhab.core.io.transport.modbus.AsyncModbusFailure; +import org.openhab.core.io.transport.modbus.AsyncModbusReadResult; +import org.openhab.core.io.transport.modbus.ModbusBitUtilities; +import org.openhab.core.io.transport.modbus.ModbusConstants; +import org.openhab.core.io.transport.modbus.ModbusReadFunctionCode; +import org.openhab.core.io.transport.modbus.ModbusReadRequestBlueprint; +import org.openhab.core.thing.ChannelUID; +import org.openhab.core.thing.Thing; +import org.openhab.core.thing.ThingStatus; +import org.openhab.core.thing.ThingStatusDetail; +import org.openhab.core.types.Command; +import org.openhab.core.types.RefreshType; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * The {@link SungrowInverterHandler} is responsible for reading the modbus values of the + * sungrow inverter. + * + * @author Sönke Küper - Initial contribution + */ +@NonNullByDefault +public class SungrowInverterHandler extends BaseModbusThingHandler { + + @NonNullByDefault + private static final class ModbusRequest { + + private final Deque registers; + private final ModbusReadRequestBlueprint blueprint; + + public ModbusRequest(Deque registers, int slaveId) { + this.registers = registers; + this.blueprint = initReadRequest(registers, slaveId); + } + + private ModbusReadRequestBlueprint initReadRequest(Deque registers, int slaveId) { + int firstRegister = registers.getFirst().getRegisterNumber(); + int lastRegister = registers.getLast().getRegisterNumber(); + int length = lastRegister - firstRegister + registers.getLast().getRegisterCount(); + assert length <= ModbusConstants.MAX_REGISTERS_READ_COUNT; + + return new ModbusReadRequestBlueprint( // + slaveId, // + ModbusReadFunctionCode.READ_INPUT_REGISTERS, // + firstRegister - 1, // + length, // + TRIES // + ); + } + } + + private final Logger logger = LoggerFactory.getLogger(SungrowInverterHandler.class); + + private static final int TRIES = 1; + private List modbusRequests = new ArrayList<>(); + + public SungrowInverterHandler(Thing thing) { + super(thing); + } + + /** + * Splits the SungrowInverterRegisters into multiple ModbusRequest, to ensure the max request size. + */ + private List buildRequests() { + final List requests = new ArrayList<>(); + Deque currentRequest = new ArrayDeque<>(); + int currentRequestFirstRegister = 0; + + for (SungrowInverterRegisters channel : SungrowInverterRegisters.values()) { + + if (currentRequest.isEmpty()) { + currentRequest.add(channel); + currentRequestFirstRegister = channel.getRegisterNumber(); + } else { + int sizeWithRegisterAdded = channel.getRegisterNumber() - currentRequestFirstRegister + + channel.getRegisterCount(); + if (sizeWithRegisterAdded > ModbusConstants.MAX_REGISTERS_READ_COUNT) { + requests.add(new ModbusRequest(currentRequest, getSlaveId())); + currentRequest = new ArrayDeque<>(); + + currentRequest.add(channel); + currentRequestFirstRegister = channel.getRegisterNumber(); + } else { + currentRequest.add(channel); + } + } + } + + if (!currentRequest.isEmpty()) { + requests.add(new ModbusRequest(currentRequest, getSlaveId())); + } + logger.debug("Created {} modbus request templates.", requests.size()); + return requests; + } + + @Override + public void handleCommand(ChannelUID channelUID, Command command) { + if (command instanceof RefreshType && !this.modbusRequests.isEmpty()) { + for (ModbusRequest request : this.modbusRequests) { + submitOneTimePoll( // + request.blueprint, // + (AsyncModbusReadResult result) -> this.readSuccessful(request, result), // + this::readError // + ); + } + } + } + + @Override + public void modbusInitialize() { + final SungrowInverterConfiguration config = getConfigAs(SungrowInverterConfiguration.class); + + if (config.pollInterval <= 0) { + updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, + "Invalid poll interval: " + config.pollInterval); + return; + } + + this.updateStatus(ThingStatus.UNKNOWN); + + this.modbusRequests = this.buildRequests(); + + for (ModbusRequest request : modbusRequests) { + registerRegularPoll( // + request.blueprint, // + config.pollInterval, // + 0, // + (AsyncModbusReadResult result) -> this.readSuccessful(request, result), // + this::readError // + ); + } + } + + private void readSuccessful(ModbusRequest request, AsyncModbusReadResult result) { + result.getRegisters().ifPresent(registers -> { + if (getThing().getStatus() != ThingStatus.ONLINE) { + updateStatus(ThingStatus.ONLINE); + } + + int firstRegister = request.registers.getFirst().getRegisterNumber(); + + for (SungrowInverterRegisters channel : request.registers) { + int index = channel.getRegisterNumber() - firstRegister; + + ModbusBitUtilities.extractStateFromRegisters(registers, index, channel.getType()) + .map(channel::createState).ifPresent(v -> updateState(createChannelUid(channel), v)); + } + }); + } + + private void readError(AsyncModbusFailure error) { + this.logger.debug("Failed to get modbus data", error.getCause()); + updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, + "Failed to retrieve data: " + error.getCause().getMessage()); + } + + private ChannelUID createChannelUid(SungrowInverterRegisters register) { + return new ChannelUID( // + thing.getUID(), // + "sg-" + register.getChannelGroup(), // + "sg-" + register.getChannelName() // + ); + } +} diff --git a/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/SungrowInverterRegisters.java b/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/SungrowInverterRegisters.java new file mode 100644 index 0000000000000..12517febfb7a5 --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/src/main/java/org/openhab/binding/modbus/sungrow/internal/SungrowInverterRegisters.java @@ -0,0 +1,255 @@ +/** + * Copyright (c) 2010-2024 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.modbus.sungrow.internal; + +import static org.openhab.core.io.transport.modbus.ModbusConstants.ValueType.INT16; +import static org.openhab.core.io.transport.modbus.ModbusConstants.ValueType.INT32_SWAP; +import static org.openhab.core.io.transport.modbus.ModbusConstants.ValueType.UINT16; +import static org.openhab.core.io.transport.modbus.ModbusConstants.ValueType.UINT32_SWAP; + +import java.math.BigDecimal; +import java.math.BigInteger; +import java.util.function.Function; + +import javax.measure.Unit; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.openhab.core.io.transport.modbus.ModbusConstants.ValueType; +import org.openhab.core.library.types.DecimalType; +import org.openhab.core.library.types.PercentType; +import org.openhab.core.library.types.QuantityType; +import org.openhab.core.library.unit.Units; +import org.openhab.core.types.State; + +/** + * The {@link SungrowInverterRegisters} is responsible for defining Modbus registers and their units. + * + * @author Sönke Küper - Initial contribution + */ +@NonNullByDefault +public enum SungrowInverterRegisters { + + // the following register numbers are 1-based. They need to be converted before sending them on the wire. + // Registers are duplicate to DAILY_PV_GENERATION / TOTAL_PV_GENERATION + // DAILY_OUTPUT_ENERGY(5003, UINT16, ConversionConstants.DIV_BY_TEN, quantityTypeFactory(Units.KILOWATT_HOUR), + // TOTAL_OUTPUT_ENERGY(5004, UINT32_SWAP, ConversionConstants.DIV_BY_TEN, quantityTypeFactory(Units.KILOWATT_HOUR), + + INTERNAL_TEMPERATURE(5008, INT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KELVIN), + ConversionConstants.CELSIUS_TO_KELVIN, "overview"), + MPPT1_VOLTAGE(5011, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.VOLT), "mppt-information"), + MPPT1_CURRENT(5012, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.AMPERE), "mppt-information"), + MPPT2_VOLTAGE(5013, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.VOLT), "mppt-information"), + MPPT2_CURRENT(5014, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.AMPERE), "mppt-information"), + TOTAL_DC_POWER(5017, UINT32_SWAP, BigDecimal.ONE, quantityFactory(Units.WATT), "overview"), + PHASE_A_VOLTAGE(5019, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.VOLT), "overview"), + PHASE_B_VOLTAGE(5020, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.VOLT), "overview"), + PHASE_C_VOLTAGE(5021, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.VOLT), "overview"), + + REACTIVE_POWER(5033, INT32_SWAP, BigDecimal.ONE, quantityFactory(Units.VAR), "overview"), + POWER_FACTOR(5035, INT16, new BigDecimal(BigInteger.ONE, 3), DecimalType::new, "overview"), + GRID_FREQUENCY(5036, UINT16, new BigDecimal(BigInteger.ONE, 2), quantityFactory(Units.HERTZ), "overview"), + + /* + * Not working + * EXPORT_LIMIT_MIN(10, 5622, UINT16, quantityTypeFactory(Units.WATT)), + * EXPORT_LIMIT_MAX(10, 5623, UINT16, quantityTypeFactory(Units.WATT)), + * BDC_RATED_POWER(100, 5628, UINT16, quantityTypeFactory(Units.WATT)), + * MAX_CHARGING_CURRENT(1, 5635, UINT16, quantityTypeFactory(Units.AMPERE)), + * MAX_DISCHARGING_CURRENT(1, 5636, UINT16, quantityTypeFactory(Units.AMPERE)), + * PV_POWER_TODAY(1, 6100, UINT16, quantityTypeFactory(Units.WATT)), + * DAILY_PV_ENERGY_YIELDS(1, 6196, UINT16, quantityTypeFactory(Units.KILOWATT_HOUR)), + * MONTHLY_PV_ENERGY_YIELDS(1, 9227, UINT16, quantityTypeFactory(Units.KILOWATT_HOUR)), + */ + + /** + * Registers return invalid values. + * SYSTEM_STATE(13000, UINT16, 1, quantityTypeFactory(Units.ONE)), + * RUNNING_STATE(13001, UINT16, 1, quantityTypeFactory(Units.ONE)), + */ + + DAILY_PV_GENERATION(13002, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "overview"), + TOTAL_PV_GENERATION(13003, UINT32_SWAP, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "overview"), + DAILY_EXPORT_ENERGY_FROM_PV(13005, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "grid-information"), + TOTAL_EXPORT_ENERGY_FROM_PV(13006, UINT32_SWAP, ConversionConstants.DIV_BY_TEN, + quantityFactory(Units.KILOWATT_HOUR), "grid-information"), + LOAD_POWER(13008, INT32_SWAP, BigDecimal.ONE, quantityFactory(Units.WATT), "load-information"), + EXPORT_POWER(13010, INT32_SWAP, BigDecimal.ONE, quantityFactory(Units.WATT), "grid-information"), + DAILY_BATTERY_CHARGE(13012, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "battery-information"), + TOTAL_BATTERY_CHARGE(13013, UINT32_SWAP, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "battery-information"), + /* + * Not working + * CO2_REDUCTION(13015, UINT32_SWAP, ConversionConstants.DIV_BY_TEN, tech.units.indriya.unit.Units.KILOGRAM), + */ + DAILY_DIRECT_ENERGY_CONSUMPTION(13017, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "load-information"), + TOTAL_DIRECT_ENERGY_CONSUMPTION(13018, UINT32_SWAP, ConversionConstants.DIV_BY_TEN, + quantityFactory(Units.KILOWATT_HOUR), "load-information"), + BATTERY_VOLTAGE(13020, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.VOLT), "battery-information"), + BATTERY_CURRENT(13021, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.AMPERE), + "battery-information"), + BATTERY_POWER(13022, UINT16, BigDecimal.ONE, quantityFactory(Units.WATT), "battery-information"), + BATTERY_LEVEL(13023, UINT16, ConversionConstants.DIV_BY_TEN, PercentType::new, "battery-information"), + BATTERY_HEALTHY(13024, UINT16, ConversionConstants.DIV_BY_TEN, PercentType::new, "battery-information"), + BATTERY_TEMPERATURE(13025, INT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KELVIN), + ConversionConstants.CELSIUS_TO_KELVIN, "battery-information"), + DAILY_BATTERY_DISCHARGE_ENERGY(13026, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "battery-information"), + TOTAL_BATTERY_DISCHARGE_ENERGY(13027, UINT32_SWAP, ConversionConstants.DIV_BY_TEN, + quantityFactory(Units.KILOWATT_HOUR), "battery-information"), + SELF_CONSUMPTION_TODAY(13029, UINT16, ConversionConstants.DIV_BY_TEN, PercentType::new, "load-information"), + // Not working + // GRID_STATE(13030, UINT16, 1, quantityTypeFactory(Units.ONE, "grid-information"), + PHASE_A_CURRENT(13031, INT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.AMPERE), "overview"), + PHASE_B_CURRENT(13032, INT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.AMPERE), "overview"), + PHASE_C_CURRENT(13033, INT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.AMPERE), "overview"), + TOTAL_ACTIVE_POWER(13034, INT32_SWAP, BigDecimal.ONE, quantityFactory(Units.WATT), "overview"), + DAILY_IMPORT_ENERGY(13036, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "grid-information"), + TOTAL_IMPORT_ENERGY(13037, UINT32_SWAP, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "grid-information"), + BATTERY_CAPACITY(13039, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "battery-information"), + DAILY_CHARGE_ENERGY(13040, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "battery-information"), + TOTAL_CHARGE_ENERGY(13041, UINT32_SWAP, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "battery-information"), + // DRM_STATE(13043, UINT16, 1, quantityTypeFactory(Units.ONE, "channelGroup"), + + DAILY_EXPORT_ENERGY(13045, UINT16, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "grid-information"), + TOTAL_EXPORT_ENERGY(13046, UINT32_SWAP, ConversionConstants.DIV_BY_TEN, quantityFactory(Units.KILOWATT_HOUR), + "grid-information"); + + /* + * Status Registers -not known if working so not implemented yet. + * + * + * INVERTER_ALARM(13050, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * GRID_SIDE_FAULT(13052, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * SYSTEM_FAULT_1(13054, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * SYSTEM_FAULT_2(13056, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * DC_SIDE_FAULT(13058, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * PERMANENT_FAULT(13060, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * BDC_SIDE_FAULT(13062, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * BDC_SIDE_PERMANENT_FAULT(13064, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * BATTERY_FAULT(13066, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * BATTERY_ALARM(13068, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * BMS_ALARM_1(13070, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * BMS_PROTECTION(13072, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * BMS_FAULT_1(13074, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * BMS_FAULT_2(13076, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE), + * BMS_ALARM_2(13078, UINT32_SWAP, 1, quantityTypeFactory(Units.ONE); + */ + + private final BigDecimal multiplier; + private final int registerNumber; + private final ValueType type; + + private final Function conversion; + private final Function stateFactory; + private final String channelGroup; + + SungrowInverterRegisters(int registerNumber, ValueType type, BigDecimal multiplier, + Function stateFactory, Function conversion, + String channelGroup) { + this.multiplier = multiplier; + this.registerNumber = registerNumber; + this.type = type; + this.conversion = conversion; + this.stateFactory = stateFactory; + this.channelGroup = channelGroup; + } + + SungrowInverterRegisters(int registerNumber, ValueType type, BigDecimal multiplier, + Function stateFactory, String channelGroup) { + this.multiplier = multiplier; + this.registerNumber = registerNumber; + this.type = type; + this.conversion = Function.identity(); + this.stateFactory = stateFactory; + this.channelGroup = channelGroup; + } + + /** + * Creates a Function that creates {@link QuantityType} states with the given {@link Unit}. + * + * @param unit {@link Unit} to be used for the value. + * @return Function for value creation. + */ + private static Function quantityFactory(Unit unit) { + return (BigDecimal value) -> new QuantityType<>(value, unit); + } + + /** + * Returns the modbus register number. + * + * @return modbus register number. + */ + public int getRegisterNumber() { + return registerNumber; + } + + /** + * Returns the {@link ValueType} for the channel. + * + * @return {@link ValueType} for the channel. + */ + public ValueType getType() { + return type; + } + + /** + * Returns the count of registers read to return the value of this register. + * + * @return register count. + */ + public int getRegisterCount() { + return this.type.getBits() / 16; + } + + /** + * Returns the channel group. + * + * @return channel group id. + */ + public String getChannelGroup() { + return channelGroup; + } + + /** + * Returns the channel name. + * + * @return the channel name. + */ + public String getChannelName() { + return this.name().toLowerCase().replace('_', '-'); + } + + /** + * Creates the {@link State} for the given register value. + * + * @param registerValue the value for the channel. + * @return {@link State] for the given value. + */ + public State createState(DecimalType registerValue) { + final BigDecimal scaledValue = registerValue.toBigDecimal().multiply(this.multiplier); + + final BigDecimal convertedValue = conversion.apply(scaledValue); + return this.stateFactory.apply(convertedValue); + } +} diff --git a/bundles/org.openhab.binding.modbus.sungrow/src/main/resources/OH-INF/config/config.xml b/bundles/org.openhab.binding.modbus.sungrow/src/main/resources/OH-INF/config/config.xml new file mode 100644 index 0000000000000..10a87cdb03014 --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/src/main/resources/OH-INF/config/config.xml @@ -0,0 +1,15 @@ + + + + + + + Time between polling the data in ms. + 5000 + + + + diff --git a/bundles/org.openhab.binding.modbus.sungrow/src/main/resources/OH-INF/i18n/sungrow.properties b/bundles/org.openhab.binding.modbus.sungrow/src/main/resources/OH-INF/i18n/sungrow.properties new file mode 100644 index 0000000000000..510aeeb41dd76 --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/src/main/resources/OH-INF/i18n/sungrow.properties @@ -0,0 +1,64 @@ +# thing types + +thing-type.modbus.sungrowInverter.label = Sungrow inverter +thing-type.modbus.sungrowInverter.description = Sungrow inverter connected via Modbus. + +# thing types config + +thing-type.config.sungrow.inverter.pollInterval.label = Poll Interval +thing-type.config.sungrow.inverter.pollInterval.description = Time between polling the data in ms. + +# channel groups +channel-group-type.modbus.sg-overview.label = Overview +channel-group-type.modbus.sg-mppt_information.label = MPPT +channel-group-type.modbus.sg-battery_information.label = Battery +channel-group-type.modbus.sg-load_information.label = Load +channel-group-type.modbus.sg-grid_information.label = Grid + +# channel types + +channel-type.modbus.sg-battery_capacity.label = Battery Capacity +channel-type.modbus.sg-battery_current.label = Battery Current +channel-type.modbus.sg-battery_healthy.label = Battery Health (SOH) +channel-type.modbus.sg-battery_level.label = Battery Level (SOC) +channel-type.modbus.sg-battery_power.label = Battery Power +channel-type.modbus.sg-battery_temperature.label = Battery Temperature +channel-type.modbus.sg-battery_voltage.label = Battery Voltage +channel-type.modbus.sg-co2_reduction.label = Co2 Reduction +channel-type.modbus.sg-daily_battery_charge.label = Daily Battery Charging Energy from PV +channel-type.modbus.sg-daily_battery_discharge_energy.label = Daily Battery Discharging Energy +channel-type.modbus.sg-daily_charge_energy.label = Daily Battery Charging Energy +channel-type.modbus.sg-daily_direct_energy_consumption.label = Daily Load Energy Consumption from PV +channel-type.modbus.sg-daily_export_energy.label = Daily Feed-in Energy +channel-type.modbus.sg-daily_export_power_from_pv.label = Daily Feed-in Energy (PV) +channel-type.modbus.sg-daily_import_energy.label = Daily Purchased Energy +channel-type.modbus.sg-daily_pv_generation.label = Daily PV Yield +channel-type.modbus.sg-export_power.label = Total Export Active Power +channel-type.modbus.sg-grid_frequency.label = Grid Frequency +channel-type.modbus.sg-internal_temperature.label = Internal Temperature +channel-type.modbus.sg-load_power.label = Load Power +channel-type.modbus.sg-mppt1_current.label = MPPT1 Current +channel-type.modbus.sg-mppt1_voltage.label = MPPT1 Voltage +channel-type.modbus.sg-mppt2_current.label = MPPT2 Current +channel-type.modbus.sg-mppt2_voltage.label = MPPT2 Voltage +channel-type.modbus.sg-phase_a_current.label = Phase A Current +channel-type.modbus.sg-phase_a_voltage.label = Phase A Voltage +channel-type.modbus.sg-phase_b_current.label = Phase B Current +channel-type.modbus.sg-phase_b_voltage.label = Phase B Voltage +channel-type.modbus.sg-phase_c_current.label = Phase C Current +channel-type.modbus.sg-phase_c_voltage.label = Phase C Voltage +channel-type.modbus.sg-power_factor.label = Total Power Factor +channel-type.modbus.sg-reactive_power.label = Total Reactive Power +channel-type.modbus.sg-running_state.label = Running Status +channel-type.modbus.sg-self_consumption_today.label = Daily Self-consumption Rate +channel-type.modbus.sg-system_state.label = Current Status +channel-type.modbus.sg-total_active_power.label = Total Active Power +channel-type.modbus.sg-total_battery_charge.label = Total Battery Charging Energy +channel-type.modbus.sg-total_battery_discharge_energy.label = Total Battery Discharging Energy +channel-type.modbus.sg-total_charge_energy.label = Total Battery Charging Energy from PV +channel-type.modbus.sg-total_dc_power.label = Total DC Power +channel-type.modbus.sg-total_direct_energy_consumption.label = Total Load Energy Consumption from PV +channel-type.modbus.sg-total_export_energy.label = Total Feed-in Energy +channel-type.modbus.sg-total_export_energy_from_pv.label = Total Feed-in Energy (PV) +channel-type.modbus.sg-total_import_energy.label = Total Purchased Energy +channel-type.modbus.sg-total_pv_generation.label = Total PV Yield diff --git a/bundles/org.openhab.binding.modbus.sungrow/src/main/resources/OH-INF/thing/thing-types.xml b/bundles/org.openhab.binding.modbus.sungrow/src/main/resources/OH-INF/thing/thing-types.xml new file mode 100644 index 0000000000000..127c10bdd6852 --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/src/main/resources/OH-INF/thing/thing-types.xml @@ -0,0 +1,507 @@ + + + + + + + + + + + Sungrow inverter connected via Modbus. + Inverter + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Battery + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Number:Temperature + + Temperature + + Measurement + Temperature + + + + + Number:ElectricPotential + + Energy + + Measurement + Voltage + + + + + Number:ElectricCurrent + + Energy + + Measurement + Current + + + + + Number:ElectricPotential + + Energy + + Measurement + Voltage + + + + + Number:ElectricCurrent + + Energy + + Measurement + Current + + + + + Number:Power + + Energy + + Measurement + Power + + + + + Number:ElectricPotential + + Energy + + Measurement + Voltage + + + + + Number:ElectricPotential + + Energy + + Measurement + Voltage + + + + + Number:ElectricPotential + + Energy + + Measurement + Voltage + + + + + Number:Power + + Energy + + Measurement + Power + + + + + Number:Dimensionless + + Energy + + + + Number:Frequency + + + Measurement + Frequency + + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Power + + Energy + + Measurement + Power + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Power + + Energy + + Measurement + Power + + + + + Number:Power + + Energy + + Measurement + Power + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:ElectricPotential + + Energy + + Measurement + Voltage + + + + + Number:ElectricCurrent + + Energy + + Measurement + Current + + + + + Number:Power + + Battery + + Measurement + Power + + + + + Number:Dimensionless + + Battery + + Measurement + Energy + + + + + Number:Dimensionless + + Battery + + + + Number:Temperature + + Battery + + Measurement + Temperature + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Dimensionless + + + + + Number:ElectricCurrent + + Energy + + Measurement + Current + + + + + Number:ElectricCurrent + + Energy + + Measurement + Current + + + + + Number:ElectricCurrent + + Energy + + Measurement + Current + + + + + Number:Power + + Energy + + Measurement + Power + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + + Number:Energy + + Energy + + Measurement + Energy + + + + diff --git a/bundles/org.openhab.binding.modbus.sungrow/src/test/java/org/openhab/binding/modbus/sungrow/internal/SungrowInverterRegistersTest.java b/bundles/org.openhab.binding.modbus.sungrow/src/test/java/org/openhab/binding/modbus/sungrow/internal/SungrowInverterRegistersTest.java new file mode 100644 index 0000000000000..422bcf9954c08 --- /dev/null +++ b/bundles/org.openhab.binding.modbus.sungrow/src/test/java/org/openhab/binding/modbus/sungrow/internal/SungrowInverterRegistersTest.java @@ -0,0 +1,73 @@ +/** + * Copyright (c) 2010-2024 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.modbus.sungrow.internal; + +import static org.junit.jupiter.api.Assertions.*; + +import java.math.BigDecimal; +import java.util.Optional; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.junit.jupiter.api.Test; +import org.openhab.core.io.transport.modbus.ModbusBitUtilities; +import org.openhab.core.io.transport.modbus.ModbusRegisterArray; +import org.openhab.core.library.types.DecimalType; +import org.openhab.core.library.types.PercentType; +import org.openhab.core.library.types.QuantityType; +import org.openhab.core.types.State; + +/** + * @author Sönke Küper - Initial contribution + */ +@NonNullByDefault +class SungrowInverterRegistersTest { + @Test + public void testCreatePercentTypeState() { + SungrowInverterRegisters batteryLevelRegister = SungrowInverterRegisters.BATTERY_LEVEL; + + ModbusRegisterArray registers = new ModbusRegisterArray(1000); + Optional value = ModbusBitUtilities.extractStateFromRegisters( // + registers, // + 0, // + batteryLevelRegister.getType() // + ); + assertTrue(value.isPresent()); + DecimalType decimalTypeValue = value.get(); + // Value is not scaled yet + assertEquals(BigDecimal.valueOf(1000), decimalTypeValue.toBigDecimal()); + + State state = batteryLevelRegister.createState(decimalTypeValue); + assertInstanceOf(PercentType.class, state); + assertEquals("100.0", state.toFullString()); + } + + @Test + public void testCreateQuantityTypeState() { + SungrowInverterRegisters mpttVoltage = SungrowInverterRegisters.MPPT1_VOLTAGE; + + ModbusRegisterArray registers = new ModbusRegisterArray(1234); + Optional value = ModbusBitUtilities.extractStateFromRegisters( // + registers, // + 0, // + mpttVoltage.getType() // + ); + assertTrue(value.isPresent()); + DecimalType decimalTypeValue = value.get(); + // Value is not scaled yet + assertEquals(BigDecimal.valueOf(1234), decimalTypeValue.toBigDecimal()); + + State state = mpttVoltage.createState(decimalTypeValue); + assertInstanceOf(QuantityType.class, state); + assertEquals("123.4 V", state.toFullString()); + } +} diff --git a/bundles/pom.xml b/bundles/pom.xml index 7ded2a83bd725..ee828b4d706f0 100644 --- a/bundles/pom.xml +++ b/bundles/pom.xml @@ -246,6 +246,7 @@ org.openhab.binding.modbus.e3dc org.openhab.binding.modbus.sbc org.openhab.binding.modbus.studer + org.openhab.binding.modbus.sungrow org.openhab.binding.modbus.sunspec org.openhab.binding.modbus.stiebeleltron org.openhab.binding.modbus.helioseasycontrols