mirror of
https://github.com/openhab/openhab-addons.git
synced 2025-01-25 14:55:55 +01:00
[fenecon] Initial contribution (#17174)
* Initial implementation of the FENECON Binding Signed-off-by: Philipp Schneider <philipp.schneider@nixo-soft.de>
This commit is contained in:
parent
66cfcc2f64
commit
6b2462ca22
@ -112,6 +112,7 @@
|
||||
/bundles/org.openhab.binding.exec/ @kgoderis
|
||||
/bundles/org.openhab.binding.feed/ @openhab/add-ons-maintainers
|
||||
/bundles/org.openhab.binding.feican/ @Hilbrand
|
||||
/bundles/org.openhab.binding.fenecon/ @nixoso
|
||||
/bundles/org.openhab.binding.fineoffsetweatherstation/ @Andy2003
|
||||
/bundles/org.openhab.binding.flicbutton/ @pfink
|
||||
/bundles/org.openhab.binding.fmiweather/ @ssalonen
|
||||
|
@ -551,6 +551,11 @@
|
||||
<artifactId>org.openhab.binding.feican</artifactId>
|
||||
<version>${project.version}</version>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.openhab.addons.bundles</groupId>
|
||||
<artifactId>org.openhab.binding.fenecon</artifactId>
|
||||
<version>${project.version}</version>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.openhab.addons.bundles</groupId>
|
||||
<artifactId>org.openhab.binding.fineoffsetweatherstation</artifactId>
|
||||
|
13
bundles/org.openhab.binding.fenecon/NOTICE
Normal file
13
bundles/org.openhab.binding.fenecon/NOTICE
Normal file
@ -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
|
160
bundles/org.openhab.binding.fenecon/README.md
Normal file
160
bundles/org.openhab.binding.fenecon/README.md
Normal file
@ -0,0 +1,160 @@
|
||||
# FENECON Binding
|
||||
|
||||
The FENECON Binding integrates the [FENECON energy storage system](https://fenecon.de/) device into the openHAB system via [REST-API](https://docs.fenecon.de/_/de/fems/fems-app/OEM_App_REST_JSON.html).
|
||||
|
||||
With the binding, it is possible to request status information from FENECON Home to allow you home automation decisions based on the current energy management.
|
||||
|
||||
This makes it possible, for example, to switch on other consumers such as the dishwasher or washing machine in the case of power overproduction.
|
||||
|
||||
## Supported Things
|
||||
|
||||
Currently only one Thing is supported: The `home-device` connection to the FENECON energy storage system.
|
||||
|
||||
This Binding was tested with an [FENECON HOME 10](https://fenecon.de/fenecon-home-10/) device.
|
||||
|
||||
## Discovery
|
||||
|
||||
Auto-discovery is not supported.
|
||||
|
||||
## Thing Configuration
|
||||
|
||||
The FENECON Thing only needs to be configured with the `hostname`, all other parameters are optional and prefilled with the suitable default values:
|
||||
|
||||
| Parameter | Description |
|
||||
|-----------------|----------------------------------------------------------------------------------|
|
||||
| hostname | Hostname or IP address of the FENECON device, e.g. 192.168.1.11 |
|
||||
| password | Password of the FENECON device. The password for guest access is set by default. |
|
||||
| port | Port of the FENECON device. Default: 8084 |
|
||||
| refreshInterval | Interval the device is polled in sec. Default 30 seconds |
|
||||
|
||||
## Channels
|
||||
|
||||
The FENECON binding currently only provides access to read out the values from the energy storage system.
|
||||
|
||||
| Channel | Type | Read/Write | Description |
|
||||
|-------------------------------|----------------------|------------|-----------------------------------------------------------------------------|
|
||||
| state | String | R | FENECON system state: Ok, Info, Warning or Fault |
|
||||
| last-update | DateTime | R | Last successful update via REST-API from the FENECON system |
|
||||
| ess-soc | Number:Dimensionless | R | Battery state of charge in percent |
|
||||
| charger-power | Number:Power | R | Current charger power of energy storage system in watt. |
|
||||
| discharger-power | Number:Power | R | Current discharger power of energy storage system in watt. |
|
||||
| emergency-power-mode | Switch | R | Indicates if there is grid power is off and the emergency power mode is on. |
|
||||
| production-active-power | Number:Power | R | Current active power producer load in watt. |
|
||||
| production-max-active-power | Number:Power | R | Maximum active production power in watt that was measured. |
|
||||
| export-to-grid-power | Number:Power | R | Current export power to grid in watt. |
|
||||
| exported-to-grid-energy | Number:Energy | R | Total energy exported to the grid in watt per hour. |
|
||||
| consumption-active-power | Number:Power | R | Current active power consumer load in watt. |
|
||||
| consumption-max-active-power | Number:Power | R | Maximum active consumption power in watt that was measured. |
|
||||
| consumption-active-power-l1 | Number:Power | R | Current active power consumer load in watt on phase 1. |
|
||||
| consumption-active-power-l2 | Number:Power | R | Current active power consumer load in watt on phase 2. |
|
||||
| consumption-active-power-l3 | Number:Power | R | Current active power consumer load in watt on phase 3. |
|
||||
| import-from-grid-power | Number:Power | R | Current import power from grid in watt. |
|
||||
| imported-from-grid-energy | Number:Energy | R | Total energy imported from the grid in watt per hour. |
|
||||
|
||||
## Full Example
|
||||
|
||||
### fenecon.things
|
||||
|
||||
```java
|
||||
Thing fenecon:home-device:local "FENECON Home" [hostname="192.168.1.11", refreshInterval=5]
|
||||
```
|
||||
|
||||
### demo.items
|
||||
|
||||
```java
|
||||
// Sitemap Items
|
||||
Group Home "MyHome" <house> ["Indoor"]
|
||||
Group GF "GroundFloor" <groundfloor> (Home) ["GroundFloor"]
|
||||
// Utility room
|
||||
Group GF_UtilityRoom "Utility room" <energy> (Home, GF) ["Room"]
|
||||
Group GF_UtilityRoomSolar "Utility room solar" <solarplant> (GF_UtilityRoom) ["Inverter"]
|
||||
|
||||
// FENECON items
|
||||
String EssState <text> (GF_UtilityRoomSolar) ["Status"] {channel="fenecon:home-device:local:state"}
|
||||
DateTime LastFeneconUpdate <time> (GF_UtilityRoomSolar) ["Status"] {channel="fenecon:home-device:local:last-update"}
|
||||
Number:Dimensionless EssSoc <batterylevel> (GF_UtilityRoomSolar) ["Measurement"] {unit="%", channel="fenecon:home-device:local:ess-soc"}
|
||||
Number:Power ChargerPower <energy> (GF_UtilityRoomSolar) ["Measurement", "Power"] {channel="fenecon:home-device:local:charger-power"}
|
||||
Number:Power DischargerPower <energy> (GF_UtilityRoomSolar) ["Measurement", "Power"] {channel="fenecon:home-device:local:discharger-power"}
|
||||
Switch EmergencyPowerMode <switch> (GF_UtilityRoomSolar) ["Switch"] {channel="fenecon:home-device:local:emergency-power-mode"}
|
||||
|
||||
Number:Power ProductionActivePower <energy> (GF_UtilityRoomSolar) ["Measurement", "Power"] {channel="fenecon:home-device:local:production-active-power"}
|
||||
Number:Power ProductionMaxActivePower <energy> (GF_UtilityRoomSolar) ["Measurement", "Power"] {channel="fenecon:home-device:local:production-max-active-power"}
|
||||
Number:Power SellToGridPower <energy> (GF_UtilityRoomSolar) ["Measurement", "Power"] {channel="fenecon:home-device:local:export-to-grid-power"}
|
||||
Number:Energy TotalSellEnergy <energy> (GF_UtilityRoomSolar) ["Measurement", "Energy"] {channel="fenecon:home-device:local:exported-to-grid-energy"}
|
||||
|
||||
Number:Power ConsumptionActivePower <energy> (GF_UtilityRoomSolar) ["Measurement", "Power"] {channel="fenecon:home-device:local:consumption-active-power"}
|
||||
Number:Power ConsumptionMaxActivePower <energy> (GF_UtilityRoomSolar) ["Measurement", "Power"] {channel="fenecon:home-device:local:consumption-max-active-power"}
|
||||
Number:Power ConsumptionActivePowerPhase1 <energy> (GF_UtilityRoomSolar) ["Measurement", "Power"] {channel="fenecon:home-device:local:consumption-active-power-l1"}
|
||||
Number:Power ConsumptionActivePowerPhase2 <energy> (GF_UtilityRoomSolar) ["Measurement", "Power"] {channel="fenecon:home-device:local:consumption-active-power-l2"}
|
||||
Number:Power ConsumptionActivePowerPhase3 <energy> (GF_UtilityRoomSolar) ["Measurement", "Power"] {channel="fenecon:home-device:local:consumption-active-power-l3"}
|
||||
Number:Power BuyFromGridPower <energy> (GF_UtilityRoomSolar) ["Measurement", "Power"] {channel="fenecon:home-device:local:import-from-grid-power"}
|
||||
Number:Energy TotalBuyEnergy <energy> (GF_UtilityRoomSolar) ["Measurement", "Energy"] {channel="fenecon:home-device:local:imported-from-grid-energy"}
|
||||
|
||||
// Examples of items for calculating the energy purchased and sold. Look at the demo.rules section.
|
||||
Number:Currency SoldEnergy "Total sold energy [%.2f €]" <price> (GF_UtilityRoomSolar)
|
||||
Number:Currency PurchasedEnergy "Total purchased energy [%.2f €]" <price> (GF_UtilityRoomSolar)
|
||||
|
||||
```
|
||||
|
||||
### demo.sitemap
|
||||
|
||||
```perl
|
||||
sitemap demo label="FENECON Example Sitemap" {
|
||||
Frame label="Groundfloor" icon="groundfloor" {
|
||||
Group item=GF_UtilityRoom
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### demo.rules
|
||||
|
||||
:::: tabs
|
||||
|
||||
::: tab DSL
|
||||
|
||||
```java
|
||||
rule "Blackout detection"
|
||||
when
|
||||
Item EmergencyPowerMode changed to ON
|
||||
then
|
||||
val msg = "🚨 Power blackout detected, emergency power mode running."
|
||||
logInfo("PowerBlackout", msg)
|
||||
sendBroadcastNotification(msg)
|
||||
end
|
||||
|
||||
rule "Battery 100 percent"
|
||||
when
|
||||
Item EssSoc changed
|
||||
then
|
||||
var batteryState = (EssSoc.getState() as Number).intValue()
|
||||
if(batteryState == 100){
|
||||
val msg = "🔋 Full battery, consumers can be activated."
|
||||
logInfo("FullBattery", msg)
|
||||
sendBroadcastNotification(msg)
|
||||
}
|
||||
end
|
||||
|
||||
rule "Calculation sold energy"
|
||||
when
|
||||
Item TotalSellEnergy changed
|
||||
then
|
||||
val sellingPricePerKiloWattHour = 0.07 // €
|
||||
var current = (TotalSellEnergy.getState() as Number).intValue()
|
||||
var result = current * sellingPricePerKiloWattHour;
|
||||
SoldEnergy.postUpdate(result)
|
||||
end
|
||||
|
||||
rule "Calculation purchased energy"
|
||||
when
|
||||
Item TotalBuyEnergy changed
|
||||
then
|
||||
val purchasedPricePerKiloWattHour = 0.32 // €
|
||||
var current = (TotalBuyEnergy.getState() as Number).intValue()
|
||||
var result = current * purchasedPricePerKiloWattHour;
|
||||
PurchasedEnergy.postUpdate(result)
|
||||
end
|
||||
```
|
||||
|
||||
:::
|
||||
|
||||
::::
|
17
bundles/org.openhab.binding.fenecon/pom.xml
Normal file
17
bundles/org.openhab.binding.fenecon/pom.xml
Normal file
@ -0,0 +1,17 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<parent>
|
||||
<groupId>org.openhab.addons.bundles</groupId>
|
||||
<artifactId>org.openhab.addons.reactor.bundles</artifactId>
|
||||
<version>4.3.0-SNAPSHOT</version>
|
||||
</parent>
|
||||
|
||||
<artifactId>org.openhab.binding.fenecon</artifactId>
|
||||
|
||||
<name>openHAB Add-ons :: Bundles :: FENECON Binding</name>
|
||||
|
||||
</project>
|
@ -0,0 +1,9 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<features name="org.openhab.binding.fenecon-${project.version}" xmlns="http://karaf.apache.org/xmlns/features/v1.4.0">
|
||||
<repository>mvn:org.openhab.core.features.karaf/org.openhab.core.features.karaf.openhab-core/${ohc.version}/xml/features</repository>
|
||||
|
||||
<feature name="openhab-binding-fenecon" description="FENECON Binding" version="${project.version}">
|
||||
<feature>openhab-runtime-base</feature>
|
||||
<bundle start-level="80">mvn:org.openhab.addons.bundles/org.openhab.binding.fenecon/${project.version}</bundle>
|
||||
</feature>
|
||||
</features>
|
@ -0,0 +1,75 @@
|
||||
/**
|
||||
* 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.fenecon.internal;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.openhab.core.thing.ThingTypeUID;
|
||||
|
||||
/**
|
||||
* The {@link FeneconBindingConstants} class defines common constants, which are
|
||||
* used across the whole binding.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class FeneconBindingConstants {
|
||||
|
||||
private static final String BINDING_ID = "fenecon";
|
||||
|
||||
// List of all Thing Type UIDs
|
||||
public static final ThingTypeUID THING_TYPE_HOME_DEVICE = new ThingTypeUID(BINDING_ID, "home-device");
|
||||
|
||||
// List of all FENECON Addresses
|
||||
public static final String STATE_ADDRESS = "_sum/State";
|
||||
public static final String ESS_SOC_ADDRESS = "_sum/EssSoc";
|
||||
public static final String CONSUMPTION_ACTIVE_POWER_ADDRESS = "_sum/ConsumptionActivePower";
|
||||
public static final String CONSUMPTION_ACTIVE_POWER_PHASE1_ADDRESS = "_sum/ConsumptionActivePowerL1";
|
||||
public static final String CONSUMPTION_ACTIVE_POWER_PHASE2_ADDRESS = "_sum/ConsumptionActivePowerL2";
|
||||
public static final String CONSUMPTION_ACTIVE_POWER_PHASE3_ADDRESS = "_sum/ConsumptionActivePowerL3";
|
||||
public static final String CONSUMPTION_MAX_ACTIVE_POWER_ADDRESS = "_sum/ConsumptionMaxActivePower";
|
||||
public static final String PRODUCTION_MAX_ACTIVE_POWER_ADDRESS = "_sum/ProductionMaxActivePower";
|
||||
public static final String PRODUCTION_ACTIVE_POWER_ADDRESS = "_sum/ProductionActivePower";
|
||||
public static final String GRID_ACTIVE_POWER_ADDRESS = "_sum/GridActivePower";
|
||||
public static final String ESS_DISCHARGE_POWER_ADDRESS = "_sum/EssDischargePower";
|
||||
public static final String GRID_MODE_ADDRESS = "_sum/GridMode";
|
||||
public static final String GRID_SELL_ACTIVE_ENERGY_ADDRESS = "_sum/GridSellActiveEnergy";
|
||||
public static final String GRID_BUY_ACTIVE_ENERGY_ADDRESS = "_sum/GridBuyActiveEnergy";
|
||||
// Group of all FENECON Addresses
|
||||
public static final List<String> ADDRESSES = List.of(STATE_ADDRESS, GRID_MODE_ADDRESS,
|
||||
CONSUMPTION_ACTIVE_POWER_ADDRESS, CONSUMPTION_ACTIVE_POWER_PHASE1_ADDRESS,
|
||||
CONSUMPTION_ACTIVE_POWER_PHASE2_ADDRESS, CONSUMPTION_ACTIVE_POWER_PHASE3_ADDRESS,
|
||||
CONSUMPTION_MAX_ACTIVE_POWER_ADDRESS, PRODUCTION_MAX_ACTIVE_POWER_ADDRESS, PRODUCTION_ACTIVE_POWER_ADDRESS,
|
||||
GRID_ACTIVE_POWER_ADDRESS, GRID_BUY_ACTIVE_ENERGY_ADDRESS, GRID_SELL_ACTIVE_ENERGY_ADDRESS, ESS_SOC_ADDRESS,
|
||||
ESS_DISCHARGE_POWER_ADDRESS);
|
||||
|
||||
// List of all Channel IDs
|
||||
public static final String STATE_CHANNEL = "state";
|
||||
public static final String ESS_SOC_CHANNEL = "ess-soc";
|
||||
public static final String CONSUMPTION_ACTIVE_POWER_CHANNEL = "consumption-active-power";
|
||||
public static final String CONSUMPTION_ACTIVE_POWER_PHASE1_CHANNEL = "consumption-active-power-l1";
|
||||
public static final String CONSUMPTION_ACTIVE_POWER_PHASE2_CHANNEL = "consumption-active-power-l2";
|
||||
public static final String CONSUMPTION_ACTIVE_POWER_PHASE3_CHANNEL = "consumption-active-power-l3";
|
||||
public static final String CONSUMPTION_MAX_ACTIVE_POWER_CHANNEL = "consumption-max-active-power";
|
||||
public static final String PRODUCTION_MAX_ACTIVE_POWER_CHANNEL = "production-max-active-power";
|
||||
public static final String PRODUCTION_ACTIVE_POWER_CHANNEL = "production-active-power";
|
||||
public static final String EXPORT_TO_GRID_POWER_CHANNEL = "export-to-grid-power";
|
||||
public static final String IMPORT_FROM_GRID_POWER_CHANNEL = "import-from-grid-power";
|
||||
public static final String ESS_CHARGER_POWER_CHANNEL = "charger-power";
|
||||
public static final String ESS_DISCHARGER_POWER_CHANNEL = "discharger-power";
|
||||
public static final String EMERGENCY_POWER_MODE_CHANNEL = "emergency-power-mode";
|
||||
public static final String EXPORTED_TO_GRID_ENERGY_CHANNEL = "exported-to-grid-energy";
|
||||
public static final String IMPORTED_FROM_GRID_ENERGY_CHANNEL = "imported-from-grid-energy";
|
||||
public static final String LAST_UPDATE_CHANNEL = "last-update";
|
||||
}
|
@ -0,0 +1,29 @@
|
||||
/**
|
||||
* 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.fenecon.internal;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
|
||||
/**
|
||||
* The {@link FeneconConfiguration} class contains fields mapping thing configuration parameters.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class FeneconConfiguration {
|
||||
|
||||
public String hostname = "";
|
||||
public String password = "user";
|
||||
public int port = 8084;
|
||||
public int refreshInterval = 30;
|
||||
}
|
@ -0,0 +1,189 @@
|
||||
/**
|
||||
* 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.fenecon.internal;
|
||||
|
||||
import java.util.Optional;
|
||||
import java.util.concurrent.ScheduledFuture;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.eclipse.jdt.annotation.Nullable;
|
||||
import org.eclipse.jetty.client.HttpClient;
|
||||
import org.openhab.binding.fenecon.internal.api.BatteryPower;
|
||||
import org.openhab.binding.fenecon.internal.api.FeneconController;
|
||||
import org.openhab.binding.fenecon.internal.api.FeneconResponse;
|
||||
import org.openhab.binding.fenecon.internal.api.GridPower;
|
||||
import org.openhab.binding.fenecon.internal.api.State;
|
||||
import org.openhab.binding.fenecon.internal.exception.FeneconException;
|
||||
import org.openhab.core.library.types.DateTimeType;
|
||||
import org.openhab.core.library.types.OnOffType;
|
||||
import org.openhab.core.library.types.QuantityType;
|
||||
import org.openhab.core.library.types.StringType;
|
||||
import org.openhab.core.library.unit.Units;
|
||||
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.thing.binding.BaseThingHandler;
|
||||
import org.openhab.core.types.Command;
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
/**
|
||||
* The {@link FeneconHandler} is responsible for handling commands, which are
|
||||
* sent to one of the channels.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class FeneconHandler extends BaseThingHandler {
|
||||
|
||||
private final Logger logger = LoggerFactory.getLogger(FeneconHandler.class);
|
||||
|
||||
private FeneconConfiguration config = new FeneconConfiguration();
|
||||
private @Nullable ScheduledFuture<?> pollingJob;
|
||||
private @Nullable FeneconController feneconController;
|
||||
private final HttpClient httpClient;
|
||||
|
||||
public FeneconHandler(Thing thing, HttpClient httpClient) {
|
||||
super(thing);
|
||||
this.httpClient = httpClient;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void initialize() {
|
||||
config = getConfigAs(FeneconConfiguration.class);
|
||||
|
||||
updateStatus(ThingStatus.UNKNOWN);
|
||||
|
||||
feneconController = new FeneconController(config, this.httpClient);
|
||||
pollingJob = scheduler.scheduleWithFixedDelay(this::pollingCode, 0, config.refreshInterval, TimeUnit.SECONDS);
|
||||
}
|
||||
|
||||
private void pollingCode() {
|
||||
for (String eachChannel : FeneconBindingConstants.ADDRESSES) {
|
||||
try {
|
||||
@SuppressWarnings("null")
|
||||
Optional<FeneconResponse> response = feneconController.requestChannel(eachChannel);
|
||||
|
||||
if (response.isPresent()) {
|
||||
processDataPoint(response.get());
|
||||
}
|
||||
|
||||
updateStatus(ThingStatus.ONLINE);
|
||||
} catch (FeneconException err) {
|
||||
logger.trace("FENECON - connection problem on FENECON channel {}", eachChannel, err);
|
||||
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.OFFLINE.COMMUNICATION_ERROR, err.getMessage());
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
// Set last successful update cycle
|
||||
updateState(FeneconBindingConstants.LAST_UPDATE_CHANNEL, new DateTimeType());
|
||||
}
|
||||
|
||||
private void processDataPoint(FeneconResponse response) throws FeneconException {
|
||||
switch (response.address()) {
|
||||
case FeneconBindingConstants.STATE_ADDRESS:
|
||||
// {"address":"_sum/State","type":"INTEGER","accessMode":"RO","text":"0:Ok, 1:Info, 2:Warning,
|
||||
// 3:Fault","unit":"","value":0}
|
||||
State state = State.get(response);
|
||||
updateState(FeneconBindingConstants.STATE_CHANNEL, new StringType(state.state()));
|
||||
break;
|
||||
case FeneconBindingConstants.ESS_SOC_ADDRESS:
|
||||
updateState(FeneconBindingConstants.ESS_SOC_CHANNEL,
|
||||
new QuantityType<>(Integer.valueOf(response.value()), Units.PERCENT));
|
||||
break;
|
||||
case FeneconBindingConstants.CONSUMPTION_ACTIVE_POWER_ADDRESS:
|
||||
updateState(FeneconBindingConstants.CONSUMPTION_ACTIVE_POWER_CHANNEL,
|
||||
new QuantityType<>(Integer.valueOf(response.value()), Units.WATT));
|
||||
break;
|
||||
case FeneconBindingConstants.CONSUMPTION_ACTIVE_POWER_PHASE1_ADDRESS:
|
||||
updateState(FeneconBindingConstants.CONSUMPTION_ACTIVE_POWER_PHASE1_CHANNEL,
|
||||
new QuantityType<>(Integer.valueOf(response.value()), Units.WATT));
|
||||
break;
|
||||
case FeneconBindingConstants.CONSUMPTION_ACTIVE_POWER_PHASE2_ADDRESS:
|
||||
updateState(FeneconBindingConstants.CONSUMPTION_ACTIVE_POWER_PHASE2_CHANNEL,
|
||||
new QuantityType<>(Integer.valueOf(response.value()), Units.WATT));
|
||||
break;
|
||||
case FeneconBindingConstants.CONSUMPTION_ACTIVE_POWER_PHASE3_ADDRESS:
|
||||
updateState(FeneconBindingConstants.CONSUMPTION_ACTIVE_POWER_PHASE3_CHANNEL,
|
||||
new QuantityType<>(Integer.valueOf(response.value()), Units.WATT));
|
||||
break;
|
||||
case FeneconBindingConstants.CONSUMPTION_MAX_ACTIVE_POWER_ADDRESS:
|
||||
updateState(FeneconBindingConstants.CONSUMPTION_MAX_ACTIVE_POWER_CHANNEL,
|
||||
new QuantityType<>(Integer.valueOf(response.value()), Units.WATT));
|
||||
break;
|
||||
case FeneconBindingConstants.PRODUCTION_MAX_ACTIVE_POWER_ADDRESS:
|
||||
updateState(FeneconBindingConstants.PRODUCTION_MAX_ACTIVE_POWER_CHANNEL,
|
||||
new QuantityType<>(Integer.valueOf(response.value()), Units.WATT));
|
||||
break;
|
||||
case FeneconBindingConstants.PRODUCTION_ACTIVE_POWER_ADDRESS:
|
||||
updateState(FeneconBindingConstants.PRODUCTION_ACTIVE_POWER_CHANNEL,
|
||||
new QuantityType<>(Integer.valueOf(response.value()), Units.WATT));
|
||||
break;
|
||||
case FeneconBindingConstants.GRID_ACTIVE_POWER_ADDRESS:
|
||||
// Grid exchange power. Negative values for sell-to-grid; positive for buy-from-grid"
|
||||
GridPower gridPower = GridPower.get(response);
|
||||
|
||||
updateState(FeneconBindingConstants.EXPORT_TO_GRID_POWER_CHANNEL,
|
||||
new QuantityType<>(gridPower.sellTo(), Units.WATT));
|
||||
updateState(FeneconBindingConstants.IMPORT_FROM_GRID_POWER_CHANNEL,
|
||||
new QuantityType<>(gridPower.buyFrom(), Units.WATT));
|
||||
break;
|
||||
case FeneconBindingConstants.ESS_DISCHARGE_POWER_ADDRESS:
|
||||
// Actual AC-side battery discharge power of Energy Storage System.
|
||||
// Negative values for charge; positive for discharge
|
||||
BatteryPower batteryPower = BatteryPower.get(response);
|
||||
|
||||
updateState(FeneconBindingConstants.ESS_CHARGER_POWER_CHANNEL,
|
||||
new QuantityType<>(batteryPower.chargerPower(), Units.WATT));
|
||||
updateState(FeneconBindingConstants.ESS_DISCHARGER_POWER_CHANNEL,
|
||||
new QuantityType<>(batteryPower.dischargerPower(), Units.WATT));
|
||||
break;
|
||||
case FeneconBindingConstants.GRID_MODE_ADDRESS:
|
||||
// text":"1:On-Grid, 2:Off-Grid","unit":"","value":1
|
||||
Integer gridMod = Integer.valueOf(response.value());
|
||||
updateState(FeneconBindingConstants.EMERGENCY_POWER_MODE_CHANNEL,
|
||||
gridMod == 2 ? OnOffType.ON : OnOffType.OFF);
|
||||
break;
|
||||
case FeneconBindingConstants.GRID_SELL_ACTIVE_ENERGY_ADDRESS:
|
||||
// {"address":"_sum/GridSellActiveEnergy","type":"LONG","accessMode":"RO","text":"","unit":"Wh_Σ","value":374242}
|
||||
updateState(FeneconBindingConstants.EXPORTED_TO_GRID_ENERGY_CHANNEL,
|
||||
new QuantityType<>(Integer.valueOf(response.value()), Units.WATT_HOUR));
|
||||
break;
|
||||
case FeneconBindingConstants.GRID_BUY_ACTIVE_ENERGY_ADDRESS:
|
||||
// "address":"_sum/GridBuyActiveEnergy","type":"LONG","accessMode":"RO","text":"","unit":"Wh_Σ","value":1105}
|
||||
updateState(FeneconBindingConstants.IMPORTED_FROM_GRID_ENERGY_CHANNEL,
|
||||
new QuantityType<>(Integer.valueOf(response.value()), Units.WATT_HOUR));
|
||||
break;
|
||||
default:
|
||||
logger.trace("FENECON - No channel ID to address {} found.", response.address());
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void dispose() {
|
||||
ScheduledFuture<?> job = pollingJob;
|
||||
if (job != null) {
|
||||
job.cancel(true);
|
||||
pollingJob = null;
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void handleCommand(ChannelUID channelUID, Command command) {
|
||||
// Noop
|
||||
}
|
||||
}
|
@ -0,0 +1,64 @@
|
||||
/**
|
||||
* 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.fenecon.internal;
|
||||
|
||||
import static org.openhab.binding.fenecon.internal.FeneconBindingConstants.THING_TYPE_HOME_DEVICE;
|
||||
|
||||
import java.util.Set;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.eclipse.jdt.annotation.Nullable;
|
||||
import org.openhab.core.io.net.http.HttpClientFactory;
|
||||
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.Activate;
|
||||
import org.osgi.service.component.annotations.Component;
|
||||
import org.osgi.service.component.annotations.Reference;
|
||||
|
||||
/**
|
||||
* The {@link FeneconHandlerFactory} is responsible for creating things and thing
|
||||
* handlers.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
@Component(configurationPid = "binding.fenecon", service = ThingHandlerFactory.class)
|
||||
public class FeneconHandlerFactory extends BaseThingHandlerFactory {
|
||||
|
||||
private static final Set<ThingTypeUID> SUPPORTED_THING_TYPES_UIDS = Set.of(THING_TYPE_HOME_DEVICE);
|
||||
private final HttpClientFactory httpClientFactory;
|
||||
|
||||
@Activate
|
||||
public FeneconHandlerFactory(@Reference HttpClientFactory httpClientFactory) {
|
||||
this.httpClientFactory = httpClientFactory;
|
||||
}
|
||||
|
||||
@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_HOME_DEVICE.equals(thingTypeUID)) {
|
||||
return new FeneconHandler(thing, httpClientFactory.getCommonHttpClient());
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
@ -0,0 +1,39 @@
|
||||
/**
|
||||
* 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.fenecon.internal.api;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
|
||||
/**
|
||||
* The {@link BatteryPower} is a small helper class to convert the power value from battery.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public record BatteryPower(int chargerPower, int dischargerPower) {
|
||||
|
||||
public static BatteryPower get(FeneconResponse response) {
|
||||
// Actual AC-side battery discharge power of Energy Storage System.
|
||||
// Negative values for charge; positive for discharge
|
||||
Integer powerValue = Integer.valueOf(response.value());
|
||||
int chargerPower = 0;
|
||||
int dischargerPower = 0;
|
||||
if (powerValue < 0) {
|
||||
chargerPower = powerValue * -1;
|
||||
} else {
|
||||
dischargerPower = powerValue;
|
||||
}
|
||||
|
||||
return new BatteryPower(chargerPower, dischargerPower);
|
||||
}
|
||||
}
|
@ -0,0 +1,125 @@
|
||||
/**
|
||||
* 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.fenecon.internal.api;
|
||||
|
||||
import java.net.URI;
|
||||
import java.net.URISyntaxException;
|
||||
import java.util.Optional;
|
||||
import java.util.concurrent.ExecutionException;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
import java.util.concurrent.TimeoutException;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.eclipse.jetty.client.HttpClient;
|
||||
import org.eclipse.jetty.client.api.AuthenticationStore;
|
||||
import org.eclipse.jetty.client.api.ContentResponse;
|
||||
import org.eclipse.jetty.client.api.Request;
|
||||
import org.eclipse.jetty.client.util.BasicAuthentication;
|
||||
import org.eclipse.jetty.http.HttpMethod;
|
||||
import org.openhab.binding.fenecon.internal.FeneconConfiguration;
|
||||
import org.openhab.binding.fenecon.internal.exception.FeneconAuthenticationException;
|
||||
import org.openhab.binding.fenecon.internal.exception.FeneconCommunicationException;
|
||||
import org.openhab.binding.fenecon.internal.exception.FeneconException;
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import com.google.gson.JsonObject;
|
||||
import com.google.gson.JsonParser;
|
||||
import com.google.gson.JsonSyntaxException;
|
||||
|
||||
/**
|
||||
* The {@link FeneconController} class provides API access to the FENECON system.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class FeneconController {
|
||||
|
||||
private final Logger logger = LoggerFactory.getLogger(FeneconController.class);
|
||||
|
||||
private final FeneconConfiguration config;
|
||||
private final HttpClient httpClient;
|
||||
|
||||
public FeneconController(FeneconConfiguration config, HttpClient httpClient) {
|
||||
this.config = config;
|
||||
this.httpClient = httpClient;
|
||||
|
||||
logger.debug("FENECON: initialize REST-API connection to {} with polling interval: {} sec", getBaseUrl(config),
|
||||
config.refreshInterval);
|
||||
|
||||
// Set BasicAuthentication for all requests on the http connection
|
||||
AuthenticationStore auth = httpClient.getAuthenticationStore();
|
||||
URI uri = URI.create(getBaseUrl(config));
|
||||
auth.addAuthenticationResult(new BasicAuthentication.BasicResult(uri, "x", config.password));
|
||||
}
|
||||
|
||||
private String getBaseUrl(FeneconConfiguration config) {
|
||||
return "http://" + config.hostname + ":" + config.port + "/";
|
||||
}
|
||||
|
||||
/**
|
||||
* Queries the data for a specified channel.
|
||||
*
|
||||
* @param channel Channel to be queried, e.g. _sum/State .
|
||||
* @return {@link FeneconResponse} can be optional if values are not available.
|
||||
* @throws FeneconException is thrown if there are problems with the connection or processing of data to the FENECON
|
||||
* system.
|
||||
*/
|
||||
public Optional<FeneconResponse> requestChannel(String channel) throws FeneconException {
|
||||
try {
|
||||
URI uri = new URI(getBaseUrl(config) + "rest/channel/" + channel);
|
||||
|
||||
Request request = httpClient.newRequest(uri).timeout(10, TimeUnit.SECONDS).method(HttpMethod.GET);
|
||||
logger.trace("FENECON - request: {}", request);
|
||||
|
||||
ContentResponse response = request.send();
|
||||
logger.trace("FENECON - response status code: {} body: {}", response.getStatus(),
|
||||
response.getContentAsString());
|
||||
|
||||
int statusCode = response.getStatus();
|
||||
if (statusCode > 300) {
|
||||
// Authentication error
|
||||
if (statusCode == 401) {
|
||||
throw new FeneconAuthenticationException(
|
||||
"Authentication on the FENECON system was not possible. Check password.");
|
||||
} else {
|
||||
throw new FeneconCommunicationException("Unexpected http status code: " + statusCode);
|
||||
}
|
||||
} else {
|
||||
return createResponseFromJson(JsonParser.parseString(response.getContentAsString()).getAsJsonObject());
|
||||
}
|
||||
} catch (TimeoutException | ExecutionException | UnsupportedOperationException | InterruptedException err) {
|
||||
throw new FeneconCommunicationException("Communication error with FENECON system on channel: " + channel,
|
||||
err);
|
||||
} catch (URISyntaxException | JsonSyntaxException err) {
|
||||
throw new FeneconCommunicationException("Syntax error on channel: " + channel, err);
|
||||
}
|
||||
}
|
||||
|
||||
private Optional<FeneconResponse> createResponseFromJson(JsonObject response) {
|
||||
// Example response: {"address":"_sum/EssSoc","type":"INTEGER","accessMode":"RO","text":"Range
|
||||
// 0..100","unit":"%","value":99}
|
||||
|
||||
if (response.get("value").isJsonNull()) {
|
||||
// Example problem response: {"address":"_sum/EssSoc","type":"INTEGER","accessMode":"RO","text":"Range
|
||||
// 0..100","unit":"%","value":null}
|
||||
return Optional.empty();
|
||||
}
|
||||
|
||||
String address = response.get("address").getAsString();
|
||||
String text = response.get("text").getAsString();
|
||||
String value = response.get("value").getAsString();
|
||||
|
||||
return Optional.of(new FeneconResponse(address, text, value));
|
||||
}
|
||||
}
|
@ -0,0 +1,24 @@
|
||||
/**
|
||||
* 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.fenecon.internal.api;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
|
||||
/**
|
||||
* The {@link FeneconResponse} class provides the response from the FENECON system.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public record FeneconResponse(String address, String text, String value) {
|
||||
};
|
@ -0,0 +1,38 @@
|
||||
/**
|
||||
* 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.fenecon.internal.api;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
|
||||
/**
|
||||
* The {@link GridPower} is a small helper class to convert the grid value.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public record GridPower(int sellTo, int buyFrom) {
|
||||
|
||||
public static GridPower get(FeneconResponse response) {
|
||||
// Grid exchange power. Negative values for sell-to-grid; positive for buy-from-grid"
|
||||
Integer gridValue = Integer.valueOf(response.value());
|
||||
int selltoGridPower = 0;
|
||||
int buyFromGridPower = 0;
|
||||
if (gridValue < 0) {
|
||||
selltoGridPower = gridValue * -1;
|
||||
} else {
|
||||
buyFromGridPower = gridValue;
|
||||
}
|
||||
|
||||
return new GridPower(selltoGridPower, buyFromGridPower);
|
||||
}
|
||||
}
|
@ -0,0 +1,43 @@
|
||||
/**
|
||||
* 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.fenecon.internal.api;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
|
||||
/**
|
||||
* The {@link State} is a small helper class to convert the state value.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public record State(String state) {
|
||||
|
||||
public static State get(FeneconResponse response) {
|
||||
// {"address":"_sum/State","type":"INTEGER","accessMode":"RO","text":"0:Ok, 1:Info, 2:Warning,
|
||||
// 3:Fault","unit":"","value":0}
|
||||
String text = response.text();
|
||||
int begin = text.indexOf(response.value() + ":");
|
||||
int end = text.indexOf(",", begin);
|
||||
|
||||
// No value to text mapping
|
||||
if (begin < 0) {
|
||||
return new State("Unknown");
|
||||
}
|
||||
|
||||
// Last text
|
||||
if (end < 0) {
|
||||
end = text.length();
|
||||
}
|
||||
return new State(text.substring(begin + 2, end));
|
||||
}
|
||||
}
|
@ -0,0 +1,30 @@
|
||||
/**
|
||||
* 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.fenecon.internal.exception;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
|
||||
/**
|
||||
* The class {@link FeneconAuthenticationException} is thrown if a authentication on the FENECON system is not possible.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class FeneconAuthenticationException extends FeneconException {
|
||||
|
||||
private static final long serialVersionUID = -9206453599559316730L;
|
||||
|
||||
public FeneconAuthenticationException(String message) {
|
||||
super(message);
|
||||
}
|
||||
}
|
@ -0,0 +1,34 @@
|
||||
/**
|
||||
* 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.fenecon.internal.exception;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
|
||||
/**
|
||||
* The class {@link FeneconCommunicationException} is thrown if a communication problem occurs with the FENECON system.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class FeneconCommunicationException extends FeneconException {
|
||||
|
||||
private static final long serialVersionUID = -4334759327203382902L;
|
||||
|
||||
public FeneconCommunicationException(String message) {
|
||||
super(message);
|
||||
}
|
||||
|
||||
public FeneconCommunicationException(String message, Exception exception) {
|
||||
super(message, exception);
|
||||
}
|
||||
}
|
@ -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.fenecon.internal.exception;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
|
||||
/**
|
||||
* The {@link FeneconException} class provides general exception for this binding.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class FeneconException extends Exception {
|
||||
|
||||
private static final long serialVersionUID = 4454633961827361165L;
|
||||
|
||||
public FeneconException() {
|
||||
// noop
|
||||
}
|
||||
|
||||
public FeneconException(String message) {
|
||||
super(message);
|
||||
}
|
||||
|
||||
public FeneconException(Exception exception) {
|
||||
super(exception);
|
||||
}
|
||||
|
||||
public FeneconException(String message, Exception exception) {
|
||||
super(message, exception);
|
||||
}
|
||||
}
|
@ -0,0 +1,11 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<addon:addon id="fenecon" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xmlns:addon="https://openhab.org/schemas/addon/v1.0.0"
|
||||
xsi:schemaLocation="https://openhab.org/schemas/addon/v1.0.0 https://openhab.org/schemas/addon-1.0.0.xsd">
|
||||
|
||||
<type>binding</type>
|
||||
<name>FENECON Home Binding</name>
|
||||
<description>This is the binding for FENECON Home.</description>
|
||||
<connection>local</connection>
|
||||
|
||||
</addon:addon>
|
@ -0,0 +1,57 @@
|
||||
# add-on
|
||||
|
||||
addon.fenecon.name = FENECON Home Binding
|
||||
addon.fenecon.description = This is the binding for FENECON Home.
|
||||
|
||||
# thing types
|
||||
|
||||
thing-type.fenecon.home-device.label = FENECON Home
|
||||
thing-type.fenecon.home-device.description = Thing for FENECON Home Device
|
||||
|
||||
# thing types config
|
||||
|
||||
thing-type.config.fenecon.home-device.hostname.label = Hostname
|
||||
thing-type.config.fenecon.home-device.hostname.description = Hostname or IP address of the FENECON device, e.g. 192.168.1.11
|
||||
thing-type.config.fenecon.home-device.password.label = Password
|
||||
thing-type.config.fenecon.home-device.password.description = Password to access the device. The password for guest access is set by default.
|
||||
thing-type.config.fenecon.home-device.port.label = Port
|
||||
thing-type.config.fenecon.home-device.port.description = Port of the FENECON device
|
||||
thing-type.config.fenecon.home-device.refreshInterval.label = Refresh Interval
|
||||
thing-type.config.fenecon.home-device.refreshInterval.description = Interval the device is polled in sec.
|
||||
|
||||
# channel types
|
||||
|
||||
channel-type.fenecon.charger-power.label = Charger Power
|
||||
channel-type.fenecon.charger-power.description = Current charger power of energy storage system in watt.
|
||||
channel-type.fenecon.consumption-active-power-phase.label = Consumer Power Phase
|
||||
channel-type.fenecon.consumption-active-power-phase.description = Current active power consumer load in watt on the corresponding phase.
|
||||
channel-type.fenecon.consumption-active-power.label = Consumer Power
|
||||
channel-type.fenecon.consumption-active-power.description = Current active power consumer load in watt.
|
||||
channel-type.fenecon.consumption-max-active-power.label = Consumer Max Power
|
||||
channel-type.fenecon.consumption-max-active-power.description = Maximum active consumption power in watt that was measured.
|
||||
channel-type.fenecon.discharger-power.label = Discharger Power
|
||||
channel-type.fenecon.discharger-power.description = Current discharger power of energy storage system in watt.
|
||||
channel-type.fenecon.emergency-power-mode.label = Emergency Power Mode
|
||||
channel-type.fenecon.emergency-power-mode.description = Indicates if there is no power from the grid and the emergency power mode is on.
|
||||
channel-type.fenecon.ess-soc.label = Battery State
|
||||
channel-type.fenecon.ess-soc.description = Battery state of charge in percent
|
||||
channel-type.fenecon.export-to-grid-power.label = Export Grid Power
|
||||
channel-type.fenecon.export-to-grid-power.description = Current export power to grid in watt.
|
||||
channel-type.fenecon.exported-to-grid-energy.label = Exported Grid Energy
|
||||
channel-type.fenecon.exported-to-grid-energy.description = Total energy exported to the grid in watt per hour.
|
||||
channel-type.fenecon.import-from-grid-power.label = Import Grid Power
|
||||
channel-type.fenecon.import-from-grid-power.description = Current import power from grid in watt.
|
||||
channel-type.fenecon.imported-from-grid-energy.label = Imported Grid Energy
|
||||
channel-type.fenecon.imported-from-grid-energy.description = Total energy imported from the grid in watt per hour.
|
||||
channel-type.fenecon.last-update.label = Last Update
|
||||
channel-type.fenecon.last-update.description = Last successful update via REST-API from the FENECON system
|
||||
channel-type.fenecon.production-active-power.label = Producer Power
|
||||
channel-type.fenecon.production-active-power.description = Current active power producer load in watt.
|
||||
channel-type.fenecon.production-max-active-power.label = Producer Max Power
|
||||
channel-type.fenecon.production-max-active-power.description = Maximum active production power in watt that was measured.
|
||||
channel-type.fenecon.state.label = System State
|
||||
channel-type.fenecon.state.description = FENECON system state
|
||||
channel-type.fenecon.state.state.option.OK = Ok
|
||||
channel-type.fenecon.state.state.option.INFO = Info
|
||||
channel-type.fenecon.state.state.option.WARN = Warning
|
||||
channel-type.fenecon.state.state.option.FAULT = Fault
|
@ -0,0 +1,177 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<thing:thing-descriptions bindingId="fenecon"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xmlns:thing="https://openhab.org/schemas/thing-description/v1.0.0"
|
||||
xsi:schemaLocation="https://openhab.org/schemas/thing-description/v1.0.0 https://openhab.org/schemas/thing-description-1.0.0.xsd">
|
||||
|
||||
<!-- Sample Thing Type -->
|
||||
<thing-type id="home-device">
|
||||
|
||||
<label>FENECON Home</label>
|
||||
<description>Thing for FENECON Home Device</description>
|
||||
<category>Solarplant</category>
|
||||
|
||||
<channels>
|
||||
<channel id="state" typeId="state"/>
|
||||
<channel id="last-update" typeId="last-update"/>
|
||||
<channel id="ess-soc" typeId="ess-soc"/>
|
||||
<channel id="charger-power" typeId="charger-power"/>
|
||||
<channel id="discharger-power" typeId="discharger-power"/>
|
||||
<channel id="emergency-power-mode" typeId="emergency-power-mode"/>
|
||||
<channel id="consumption-active-power" typeId="consumption-active-power"/>
|
||||
<channel id="consumption-active-power-l1" typeId="consumption-active-power-phase"/>
|
||||
<channel id="consumption-active-power-l2" typeId="consumption-active-power-phase"/>
|
||||
<channel id="consumption-active-power-l3" typeId="consumption-active-power-phase"/>
|
||||
<channel id="consumption-max-active-power" typeId="consumption-max-active-power"/>
|
||||
<channel id="production-max-active-power" typeId="production-max-active-power"/>
|
||||
<channel id="production-active-power" typeId="production-active-power"/>
|
||||
<channel id="export-to-grid-power" typeId="export-to-grid-power"/>
|
||||
<channel id="exported-to-grid-energy" typeId="exported-to-grid-energy"/>
|
||||
<channel id="import-from-grid-power" typeId="import-from-grid-power"/>
|
||||
<channel id="imported-from-grid-energy" typeId="imported-from-grid-energy"/>
|
||||
</channels>
|
||||
|
||||
<config-description>
|
||||
<parameter name="hostname" type="text" required="true">
|
||||
<context>network-address</context>
|
||||
<label>Hostname</label>
|
||||
<description>Hostname or IP address of the FENECON device, e.g. 192.168.1.11</description>
|
||||
</parameter>
|
||||
<parameter name="password" type="text">
|
||||
<context>password</context>
|
||||
<label>Password</label>
|
||||
<default>user</default>
|
||||
<description>Password to access the device. The password for guest access is set by default.</description>
|
||||
<advanced>true</advanced>
|
||||
</parameter>
|
||||
<parameter name="port" type="integer" min="1">
|
||||
<context>network-address</context>
|
||||
<label>Port</label>
|
||||
<description>Port of the FENECON device</description>
|
||||
<default>8084</default>
|
||||
<advanced>true</advanced>
|
||||
</parameter>
|
||||
<parameter name="refreshInterval" type="integer" unit="s" min="1">
|
||||
<label>Refresh Interval</label>
|
||||
<description>Interval the device is polled in sec.</description>
|
||||
<default>30</default>
|
||||
<advanced>true</advanced>
|
||||
</parameter>
|
||||
</config-description>
|
||||
</thing-type>
|
||||
|
||||
<!-- Channel Types -->
|
||||
<channel-type id="state">
|
||||
<item-type>String</item-type>
|
||||
<label>System State</label>
|
||||
<description>FENECON system state</description>
|
||||
<category>Text</category>
|
||||
<state readOnly="true" pattern="%s">
|
||||
<options>
|
||||
<option value="OK">Ok</option>
|
||||
<option value="INFO">Info</option>
|
||||
<option value="WARN">Warning</option>
|
||||
<option value="FAULT">Fault</option>
|
||||
</options>
|
||||
</state>
|
||||
</channel-type>
|
||||
<channel-type id="last-update">
|
||||
<item-type>DateTime</item-type>
|
||||
<label>Last Update</label>
|
||||
<description>Last successful update via REST-API from the FENECON system</description>
|
||||
<category>Time</category>
|
||||
<state readOnly="true"></state>
|
||||
</channel-type>
|
||||
<channel-type id="ess-soc">
|
||||
<item-type unitHint="%">Number:Dimensionless</item-type>
|
||||
<label>Battery State</label>
|
||||
<description>Battery state of charge in percent</description>
|
||||
<category>BatteryLevel</category>
|
||||
<state readOnly="true" pattern="%.0f %unit%"/>
|
||||
</channel-type>
|
||||
<channel-type id="charger-power">
|
||||
<item-type>Number:Power</item-type>
|
||||
<label>Charger Power</label>
|
||||
<description>Current charger power of energy storage system in watt.</description>
|
||||
<category>Energy</category>
|
||||
<state readOnly="true" pattern="%.0f %unit%"/>
|
||||
</channel-type>
|
||||
<channel-type id="discharger-power">
|
||||
<item-type>Number:Power</item-type>
|
||||
<label>Discharger Power</label>
|
||||
<description>Current discharger power of energy storage system in watt.</description>
|
||||
<category>Energy</category>
|
||||
<state readOnly="true" pattern="%.0f %unit%"/>
|
||||
</channel-type>
|
||||
<channel-type id="emergency-power-mode">
|
||||
<item-type>Switch</item-type>
|
||||
<label>Emergency Power Mode</label>
|
||||
<description>Indicates if there is no power from the grid and the emergency power mode is on.</description>
|
||||
<category>Switch</category>
|
||||
<state readOnly="true"/>
|
||||
</channel-type>
|
||||
<channel-type id="production-active-power">
|
||||
<item-type>Number:Power</item-type>
|
||||
<label>Producer Power</label>
|
||||
<description>Current active power producer load in watt.</description>
|
||||
<category>Energy</category>
|
||||
<state readOnly="true" pattern="%.0f %unit%"/>
|
||||
</channel-type>
|
||||
<channel-type id="export-to-grid-power">
|
||||
<item-type>Number:Power</item-type>
|
||||
<label>Export Grid Power</label>
|
||||
<description>Current export power to grid in watt.</description>
|
||||
<category>Energy</category>
|
||||
<state readOnly="true" pattern="%.0f %unit%"/>
|
||||
</channel-type>
|
||||
<channel-type id="exported-to-grid-energy">
|
||||
<item-type>Number:Energy</item-type>
|
||||
<label>Exported Grid Energy</label>
|
||||
<description>Total energy exported to the grid in watt per hour.</description>
|
||||
<category>Energy</category>
|
||||
<state readOnly="true" pattern="%.0f %unit%"/>
|
||||
</channel-type>
|
||||
<channel-type id="consumption-active-power">
|
||||
<item-type>Number:Power</item-type>
|
||||
<label>Consumer Power</label>
|
||||
<description>Current active power consumer load in watt.</description>
|
||||
<category>Energy</category>
|
||||
<state readOnly="true" pattern="%.0f %unit%"/>
|
||||
</channel-type>
|
||||
<channel-type id="consumption-active-power-phase">
|
||||
<item-type>Number:Power</item-type>
|
||||
<label>Consumer Power Phase</label>
|
||||
<description>Current active power consumer load in watt on the corresponding phase.</description>
|
||||
<category>Energy</category>
|
||||
<state readOnly="true" pattern="%.0f %unit%"/>
|
||||
</channel-type>
|
||||
<channel-type id="consumption-max-active-power">
|
||||
<item-type>Number:Power</item-type>
|
||||
<label>Consumer Max Power</label>
|
||||
<description>Maximum active consumption power in watt that was measured.</description>
|
||||
<category>Energy</category>
|
||||
<state readOnly="true" pattern="%.0f %unit%"/>
|
||||
</channel-type>
|
||||
<channel-type id="production-max-active-power">
|
||||
<item-type>Number:Power</item-type>
|
||||
<label>Producer Max Power</label>
|
||||
<description>Maximum active production power in watt that was measured.</description>
|
||||
<category>Energy</category>
|
||||
<state readOnly="true" pattern="%.0f %unit%"/>
|
||||
</channel-type>
|
||||
<channel-type id="import-from-grid-power">
|
||||
<item-type>Number:Power</item-type>
|
||||
<label>Import Grid Power</label>
|
||||
<description>Current import power from grid in watt.</description>
|
||||
<category>Energy</category>
|
||||
<state readOnly="true" pattern="%.0f %unit%"/>
|
||||
</channel-type>
|
||||
<channel-type id="imported-from-grid-energy">
|
||||
<item-type>Number:Energy</item-type>
|
||||
<label>Imported Grid Energy</label>
|
||||
<description>Total energy imported from the grid in watt per hour.</description>
|
||||
<category>Energy</category>
|
||||
<state readOnly="true" pattern="%.0f %unit%"/>
|
||||
</channel-type>
|
||||
|
||||
</thing:thing-descriptions>
|
@ -0,0 +1,48 @@
|
||||
/**
|
||||
* 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.fenecon.internal;
|
||||
|
||||
import static org.junit.jupiter.api.Assertions.*;
|
||||
|
||||
import java.lang.reflect.Field;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.junit.jupiter.api.Test;
|
||||
|
||||
/**
|
||||
* Test for {@link FeneconBindingConstants}.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class FeneconBindingConstantsTest {
|
||||
|
||||
@Test
|
||||
void checkAllAddressesAreListed() throws IllegalArgumentException, IllegalAccessException {
|
||||
List<String> findAddresses = new ArrayList<>();
|
||||
|
||||
for (Field eachDeclaredField : FeneconBindingConstants.class.getDeclaredFields()) {
|
||||
if (eachDeclaredField.getName().endsWith("_ADDRESS")) {
|
||||
String address = (String) eachDeclaredField.get(FeneconBindingConstants.class);
|
||||
if (address != null) {
|
||||
findAddresses.add(address);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
assertEquals(FeneconBindingConstants.ADDRESSES.size(), findAddresses.size());
|
||||
assertTrue(findAddresses.containsAll(FeneconBindingConstants.ADDRESSES));
|
||||
}
|
||||
}
|
@ -0,0 +1,44 @@
|
||||
/**
|
||||
* 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.fenecon.internal.api;
|
||||
|
||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.junit.jupiter.api.Test;
|
||||
import org.openhab.binding.fenecon.internal.FeneconBindingConstants;
|
||||
|
||||
/**
|
||||
* Test for {@link BatteryPower}.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class BatteryPowerTest {
|
||||
|
||||
@Test
|
||||
void testCharging() {
|
||||
BatteryPower batteryPower = BatteryPower
|
||||
.get(new FeneconResponse(FeneconBindingConstants.ESS_DISCHARGE_POWER_ADDRESS, "comment", "-1777"));
|
||||
assertEquals(1777, batteryPower.chargerPower());
|
||||
assertEquals(0, batteryPower.dischargerPower());
|
||||
}
|
||||
|
||||
@Test
|
||||
void testDischarging() {
|
||||
BatteryPower batteryPower = BatteryPower
|
||||
.get(new FeneconResponse(FeneconBindingConstants.ESS_DISCHARGE_POWER_ADDRESS, "comment", "1777"));
|
||||
assertEquals(1777, batteryPower.dischargerPower());
|
||||
assertEquals(0, batteryPower.chargerPower());
|
||||
}
|
||||
}
|
@ -0,0 +1,44 @@
|
||||
/**
|
||||
* 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.fenecon.internal.api;
|
||||
|
||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.junit.jupiter.api.Test;
|
||||
import org.openhab.binding.fenecon.internal.FeneconBindingConstants;
|
||||
|
||||
/**
|
||||
* Test for {@link GridPower}.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class GridPowerTest {
|
||||
|
||||
@Test
|
||||
void testSelling() {
|
||||
GridPower gridValue = GridPower
|
||||
.get(new FeneconResponse(FeneconBindingConstants.GRID_ACTIVE_POWER_ADDRESS, "comment", "-1777"));
|
||||
assertEquals(1777, gridValue.sellTo());
|
||||
assertEquals(0, gridValue.buyFrom());
|
||||
}
|
||||
|
||||
@Test
|
||||
void testBuying() {
|
||||
GridPower gridValue = GridPower
|
||||
.get(new FeneconResponse(FeneconBindingConstants.GRID_ACTIVE_POWER_ADDRESS, "comment", "1777"));
|
||||
assertEquals(1777, gridValue.buyFrom());
|
||||
assertEquals(0, gridValue.sellTo());
|
||||
}
|
||||
}
|
@ -0,0 +1,63 @@
|
||||
/**
|
||||
* 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.fenecon.internal.api;
|
||||
|
||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.junit.jupiter.api.Test;
|
||||
import org.openhab.binding.fenecon.internal.FeneconBindingConstants;
|
||||
|
||||
/**
|
||||
* Test for {@link State}.
|
||||
*
|
||||
* @author Philipp Schneider - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class StateTest {
|
||||
|
||||
@Test
|
||||
void testStateOk() {
|
||||
State state = State.get(
|
||||
new FeneconResponse(FeneconBindingConstants.STATE_ADDRESS, "0:Ok, 1:Info, 2:Warning, 3:Fault", "0"));
|
||||
assertEquals("Ok", state.state());
|
||||
}
|
||||
|
||||
@Test
|
||||
void testStateInfo() {
|
||||
State state = State.get(
|
||||
new FeneconResponse(FeneconBindingConstants.STATE_ADDRESS, "0:Ok, 1:Info, 2:Warning, 3:Fault", "1"));
|
||||
assertEquals("Info", state.state());
|
||||
}
|
||||
|
||||
@Test
|
||||
void testStateWarning() {
|
||||
State state = State.get(
|
||||
new FeneconResponse(FeneconBindingConstants.STATE_ADDRESS, "0:Ok, 1:Info, 2:Warning, 3:Fault", "2"));
|
||||
assertEquals("Warning", state.state());
|
||||
}
|
||||
|
||||
@Test
|
||||
void testStateFault() {
|
||||
State state = State.get(
|
||||
new FeneconResponse(FeneconBindingConstants.STATE_ADDRESS, "0:Ok, 1:Info, 2:Warning, 3:Fault", "3"));
|
||||
assertEquals("Fault", state.state());
|
||||
}
|
||||
|
||||
@Test
|
||||
void testStateUnknown() {
|
||||
State state = State.get(
|
||||
new FeneconResponse(FeneconBindingConstants.STATE_ADDRESS, "0:Ok, 1:Info, 2:Warning, 3:Fault", "4"));
|
||||
assertEquals("Unknown", state.state());
|
||||
}
|
||||
}
|
@ -145,6 +145,7 @@
|
||||
<module>org.openhab.binding.exec</module>
|
||||
<module>org.openhab.binding.feed</module>
|
||||
<module>org.openhab.binding.feican</module>
|
||||
<module>org.openhab.binding.fenecon</module>
|
||||
<module>org.openhab.binding.fineoffsetweatherstation</module>
|
||||
<module>org.openhab.binding.flicbutton</module>
|
||||
<module>org.openhab.binding.fmiweather</module>
|
||||
|
Loading…
Reference in New Issue
Block a user