diff --git a/CODEOWNERS b/CODEOWNERS index 8ffad8e51..5462def51 100644 --- a/CODEOWNERS +++ b/CODEOWNERS @@ -97,6 +97,7 @@ /bundles/org.openhab.binding.gpstracker/ @gbicskei /bundles/org.openhab.binding.gree/ @markus7017 /bundles/org.openhab.binding.groheondus/ @FlorianSW +/bundles/org.openhab.binding.haassohnpelletstove/ @chingon007 /bundles/org.openhab.binding.harmonyhub/ @digitaldan /bundles/org.openhab.binding.haywardomnilogic/ @matchews /bundles/org.openhab.binding.hdanywhere/ @kgoderis diff --git a/bom/openhab-addons/pom.xml b/bom/openhab-addons/pom.xml index 385c85c2d..10205adc0 100644 --- a/bom/openhab-addons/pom.xml +++ b/bom/openhab-addons/pom.xml @@ -471,6 +471,11 @@ org.openhab.binding.groheondus ${project.version} + + org.openhab.addons.bundles + org.openhab.binding.haassohnpelletstove + ${project.version} + org.openhab.addons.bundles org.openhab.binding.harmonyhub diff --git a/bundles/org.openhab.binding.haassohnpelletstove/NOTICE b/bundles/org.openhab.binding.haassohnpelletstove/NOTICE new file mode 100644 index 000000000..38d625e34 --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/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.haassohnpelletstove/README.md b/bundles/org.openhab.binding.haassohnpelletstove/README.md new file mode 100644 index 000000000..2d053f481 --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/README.md @@ -0,0 +1,69 @@ +# Haas Sohn Pellet Stove Binding + +The binding for Haassohnpelletstove communicates with a Haas and Sohn Pelletstove through the optional +WIFI module. More information about the WIFI module can be found here: https://www.haassohn.com/de/ihr-plus/WLAN-Funktion + +## Supported Things + +| Things | Description | Thing Type | +|--------|--------------|------------| +| haassohnpelletstove | Control of a Haas & Sohn Pellet Stove| oven| + + +## Thing Configuration + +In general two parameters are required. The IP-Address of the WIFI-Modul of the Stove in the local Network and the Access PIN of the Stove. +The PIN can be found directly at the stove under the Menue/Network/WLAN-PIN + +``` +Thing haassohnpelletstove:oven:myOven "Pelletstove" [ hostIP="192.168.0.23", hostPIN="1234"] +``` + +## Channels + +The following channels are yet supported: + + +| Channel | Type | Access| Description| +|---------|-------|-------|------------| +| power| Switch | read/write|Turn the stove on/off| +|channelIsTemp|Number:Temperature|read|Receives the actual temperature of the stove| +|channelSpTemp|Number:Temperature|read/write|Receives and sets the target temperature of the stove| +|channelMode|String|read|Receives the actual mode the stove is in like heating, cooling, error, ....| +|channelEcoMode|Switch|read/write|Turn the eco mode of the stove on/off| +|channelIngitions|Number|read|Amount of ignitions of the stove| +|channelMaintenanceIn|Number:Mass|read|States the next maintenance in kg| +|channelCleaningIn|String|read|States the next cleaning window in hours:minutes as string| +|channelConsumption|Number:Mass|read|Total consumption of the stove| +|channelOnTime|Number|read|Operation hours of the stove| + +## Full Example + +demo.items: + +``` +Number:Temperature isTemp { channel="oven:channelIsTemp" } +Number:Temperature spTemp { channel="oven:channelSpTemp" } +String mode { channel="oven:channelMode" } +Switch power { channel="oven:power" } +``` + +## Google Assistant configuration + +See also: https://www.openhab.org/docs/ecosystem/google-assistant/ + +googleassistantdemo.items + +``` +Group g_FeuerThermostat "FeuerThermostat" {ga="Thermostat" } +Number StatusFeuer "Status Feuer" (g_FeuerThermostat) { ga="thermostatMode" } +Number ZieltemperaturFeuer "ZieltemperaturFeuer" (g_FeuerThermostat) {ga="thermostatTemperatureSetpoint"} +Number TemperaturFeuer "TemperaturFeuer" (g_FeuerThermostat) {ga="thermostatTemperatureAmbient"} +``` + +## Tested Hardware + +The binding was successfully tested with the following ovens: + +- HSP 7 DIANA +- HSP6 434.08 diff --git a/bundles/org.openhab.binding.haassohnpelletstove/pom.xml b/bundles/org.openhab.binding.haassohnpelletstove/pom.xml new file mode 100644 index 000000000..d7aed0af5 --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/pom.xml @@ -0,0 +1,17 @@ + + + + 4.0.0 + + + org.openhab.addons.bundles + org.openhab.addons.reactor.bundles + 3.1.0-SNAPSHOT + + + org.openhab.binding.haassohnpelletstove + + openHAB Add-ons :: Bundles :: Haas + Sohn Pelletstove Binding + + diff --git a/bundles/org.openhab.binding.haassohnpelletstove/src/main/feature/feature.xml b/bundles/org.openhab.binding.haassohnpelletstove/src/main/feature/feature.xml new file mode 100644 index 000000000..6861266bb --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/src/main/feature/feature.xml @@ -0,0 +1,9 @@ + + + mvn:org.openhab.core.features.karaf/org.openhab.core.features.karaf.openhab-core/${ohc.version}/xml/features + + + openhab-runtime-base + mvn:org.openhab.addons.bundles/org.openhab.binding.haassohnpelletstove/${project.version} + + diff --git a/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveBindingConstants.java b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveBindingConstants.java new file mode 100644 index 000000000..e60f57d95 --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveBindingConstants.java @@ -0,0 +1,41 @@ +/** + * Copyright (c) 2010-2021 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.haassohnpelletstove.internal; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.openhab.core.thing.ThingTypeUID; + +/** + * The {@link HaasSohnpelletstoveBindingConstants} class defines common constants, which are + * used across the whole binding. + * + * @author Christian Feininger - Initial contribution + */ +@NonNullByDefault +public class HaasSohnpelletstoveBindingConstants { + + private static final String BINDING_ID = "haassohnpelletstove"; + + public static final ThingTypeUID THING_TYPE_OVEN = new ThingTypeUID(BINDING_ID, "oven"); + + public static final String CHANNELISTEMP = "channelIsTemp"; + public static final String CHANNELMODE = "channelMode"; + public static final String CHANNELSPTEMP = "channelSpTemp"; + public static final String CHANNELPOWER = "power"; + public static final String CHANNELECOMODE = "channelEcoMode"; + public static final String CHANNELIGNITIONS = "channelIgnitions"; + public static final String CHANNELMAINTENANCEIN = "channelMaintenanceIn"; + public static final String CHANNELCLEANINGIN = "channelCleaningIn"; + public static final String CHANNELCONSUMPTION = "channelConsumption"; + public static final String CHANNELONTIME = "channelOnTime"; +} diff --git a/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveConfiguration.java b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveConfiguration.java new file mode 100644 index 000000000..cfb6e3c1a --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveConfiguration.java @@ -0,0 +1,29 @@ +/** + * Copyright (c) 2010-2021 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.haassohnpelletstove.internal; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; + +/** + * The {@link HaasSohnpelletstoveConfiguration} class contains fields mapping thing configuration parameters. + * + * @author Christian Feininger - Initial contribution + */ +@NonNullByDefault +public class HaasSohnpelletstoveConfiguration { + + public @Nullable String hostIP = null; + public @Nullable String hostPIN = null; + public int refreshRate = 30; +} diff --git a/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveHandler.java b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveHandler.java new file mode 100644 index 000000000..a8c28a108 --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveHandler.java @@ -0,0 +1,318 @@ +/** + * Copyright (c) 2010-2021 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.haassohnpelletstove.internal; + +import static org.openhab.binding.haassohnpelletstove.internal.HaasSohnpelletstoveBindingConstants.*; + +import java.text.DecimalFormat; +import java.util.HashMap; +import java.util.Map; +import java.util.concurrent.ScheduledFuture; +import java.util.concurrent.TimeUnit; + +import javax.measure.Unit; +import javax.measure.quantity.Temperature; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +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.SIUnits; +import org.openhab.core.thing.Channel; +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.openhab.core.types.State; +import org.openhab.core.types.UnDefType; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * The {@link HaasSohnpelletstoveHandler} is responsible for handling commands, which are + * sent to one of the channels. + * + * @author Christian Feininger - Initial contribution + */ +@NonNullByDefault +public class HaasSohnpelletstoveHandler extends BaseThingHandler { + + private final Logger logger = LoggerFactory.getLogger(HaasSohnpelletstoveHandler.class); + + private @Nullable ScheduledFuture refreshJob; + + private HaasSohnpelletstoveConfiguration config = new HaasSohnpelletstoveConfiguration(); + boolean resultOk = false; + + private HaasSohnpelletstoveJSONCommunication serviceCommunication; + + private boolean automaticRefreshing = false; + + private Map linkedChannels = new HashMap(); + + public HaasSohnpelletstoveHandler(Thing thing) { + super(thing); + serviceCommunication = new HaasSohnpelletstoveJSONCommunication(); + } + + @Override + public void handleCommand(ChannelUID channelUID, Command command) { + if (channelUID.getId().equals(CHANNELPOWER)) { + String postData = null; + if (command.equals(OnOffType.ON)) { + postData = "{\"prg\":true}"; + } else if (command.equals(OnOffType.OFF)) { + postData = "{\"prg\":false}"; + } + if (postData != null) { + logger.debug("Executing {} command", CHANNELPOWER); + updateOvenData(postData); + } + } else if (channelUID.getId().equals(CHANNELSPTEMP)) { + if (command instanceof QuantityType) { + QuantityType value = (QuantityType) command; + + Unit unit = SIUnits.CELSIUS; + value = value.toUnit(unit); + if (value != null) { + double a = value.doubleValue(); + String postdata = "{\"sp_temp\":" + a + "}"; + logger.debug("Executing {} command", CHANNELSPTEMP); + updateOvenData(postdata); + } + } else { + logger.debug("Error. Command is the wrong type: {}", command.toString()); + } + } else if (channelUID.getId().equals(CHANNELECOMODE)) { + String postData = null; + if (command.equals(OnOffType.ON)) { + postData = "{\"eco_mode\":true}"; + } else if (command.equals(OnOffType.OFF)) { + postData = "{\"eco_mode\":false}"; + } + if (postData != null) { + logger.debug("Executing {} command", CHANNELECOMODE); + updateOvenData(postData); + } + } + } + + /** + * Calls the service to update the oven data + * + * @param postdata + */ + private boolean updateOvenData(@Nullable String postdata) { + Helper message = new Helper(); + if (serviceCommunication.updateOvenData(postdata, message, this.getThing().getUID().toString())) { + updateStatus(ThingStatus.ONLINE); + return true; + } else { + updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.OFFLINE.COMMUNICATION_ERROR, + message.getStatusDesription()); + return false; + } + } + + @Override + public void initialize() { + logger.debug("Initializing haassohnpelletstove handler for thing {}", getThing().getUID()); + config = getConfigAs(HaasSohnpelletstoveConfiguration.class); + boolean validConfig = true; + String errors = ""; + String statusDescr = null; + if (config.refreshRate < 0 && config.refreshRate > 999) { + errors += " Parameter 'refresh Rate' greater then 0 and less then 1000."; + statusDescr = "Parameter 'refresh Rate' greater then 0 and less then 1000."; + validConfig = false; + } + if (config.hostIP == null) { + errors += " Parameter 'hostIP' must be configured."; + statusDescr = "IP Address must be configured!"; + validConfig = false; + } + if (config.hostPIN == null) { + errors += " Parameter 'hostPin' must be configured."; + statusDescr = "PIN must be configured!"; + validConfig = false; + } + errors = errors.trim(); + Helper message = new Helper(); + message.setStatusDescription(statusDescr); + if (validConfig) { + serviceCommunication.setConfig(config); + if (serviceCommunication.refreshOvenConnection(message, this.getThing().getUID().toString())) { + if (updateOvenData(null)) { + updateStatus(ThingStatus.ONLINE); + updateLinkedChannels(); + } + } else { + updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, message.getStatusDesription()); + } + } else { + updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, message.getStatusDesription()); + } + } + + private void updateLinkedChannels() { + verifyLinkedChannel(CHANNELISTEMP); + verifyLinkedChannel(CHANNELMODE); + verifyLinkedChannel(CHANNELPOWER); + verifyLinkedChannel(CHANNELSPTEMP); + verifyLinkedChannel(CHANNELECOMODE); + verifyLinkedChannel(CHANNELIGNITIONS); + verifyLinkedChannel(CHANNELMAINTENANCEIN); + verifyLinkedChannel(CHANNELCLEANINGIN); + verifyLinkedChannel(CHANNELCONSUMPTION); + verifyLinkedChannel(CHANNELONTIME); + if (!linkedChannels.isEmpty()) { + updateOvenData(null); + for (Channel channel : getThing().getChannels()) { + updateChannel(channel.getUID().getId()); + } + startAutomaticRefresh(); + automaticRefreshing = true; + } + } + + private void verifyLinkedChannel(String channelID) { + if (isLinked(channelID) && !linkedChannels.containsKey(channelID)) { + linkedChannels.put(channelID, true); + } + } + + @Override + public void dispose() { + stopScheduler(); + } + + private void stopScheduler() { + ScheduledFuture job = refreshJob; + if (job != null) { + job.cancel(true); + } + refreshJob = null; + } + + /** + * Start the job refreshing the oven status + */ + private void startAutomaticRefresh() { + ScheduledFuture job = refreshJob; + if (job == null || job.isCancelled()) { + int period = config.refreshRate; + refreshJob = scheduler.scheduleWithFixedDelay(this::run, 0, period, TimeUnit.SECONDS); + } + } + + private void run() { + updateOvenData(null); + for (Channel channel : getThing().getChannels()) { + updateChannel(channel.getUID().getId()); + } + } + + @Override + public void channelLinked(ChannelUID channelUID) { + if (!automaticRefreshing) { + logger.debug("Start automatic refreshing"); + startAutomaticRefresh(); + automaticRefreshing = true; + } + verifyLinkedChannel(channelUID.getId()); + updateChannel(channelUID.getId()); + } + + @Override + public void channelUnlinked(ChannelUID channelUID) { + linkedChannels.remove(channelUID.getId()); + if (linkedChannels.isEmpty()) { + automaticRefreshing = false; + stopScheduler(); + logger.debug("Stop automatic refreshing"); + } + } + + private void updateChannel(String channelId) { + if (isLinked(channelId)) { + State state = null; + HaasSohnpelletstoveJsonDataDTO data = serviceCommunication.getOvenData(); + if (data != null) { + switch (channelId) { + case CHANNELISTEMP: + state = new QuantityType(Double.valueOf(data.getisTemp()), SIUnits.CELSIUS); + update(state, channelId); + break; + case CHANNELMODE: + state = new StringType(data.getMode()); + update(state, channelId); + break; + case CHANNELPOWER: + update(OnOffType.from(data.getPrg()), channelId); + break; + case CHANNELECOMODE: + update(OnOffType.from(data.getEcoMode()), channelId); + break; + case CHANNELSPTEMP: + state = new QuantityType(Double.valueOf(data.getspTemp()), SIUnits.CELSIUS); + update(state, channelId); + break; + case CHANNELCLEANINGIN: + String cleaning = data.getCleaningIn(); + double time = Double.parseDouble(cleaning); + time = time / 60; + DecimalFormat df = new DecimalFormat("0.00"); + state = new StringType(df.format(time)); + update(state, channelId); + break; + case CHANNELCONSUMPTION: + state = new StringType(data.getConsumption()); + update(state, channelId); + break; + case CHANNELIGNITIONS: + state = new StringType(data.getIgnitions()); + update(state, channelId); + break; + case CHANNELMAINTENANCEIN: + state = new StringType(data.getMaintenanceIn()); + update(state, channelId); + break; + case CHANNELONTIME: + state = new StringType(data.getOnTime()); + update(state, channelId); + break; + } + } + } + } + + /** + * Updates the State of the given channel + * + * @param state + * @param channelId + */ + private void update(@Nullable State state, String channelId) { + logger.debug("Update channel {} with state {}", channelId, (state == null) ? "null" : state.toString()); + + if (state != null) { + updateState(channelId, state); + + } else { + updateState(channelId, UnDefType.NULL); + } + } +} diff --git a/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveHandlerFactory.java b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveHandlerFactory.java new file mode 100644 index 000000000..165b5105c --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveHandlerFactory.java @@ -0,0 +1,54 @@ +/** + * Copyright (c) 2010-2021 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.haassohnpelletstove.internal; + +import static org.openhab.binding.haassohnpelletstove.internal.HaasSohnpelletstoveBindingConstants.THING_TYPE_OVEN; + +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 HaasSohnpelletstoveHandlerFactory} is responsible for creating things and thing + * handlers. + * + * @author Christian Feininger - Initial contribution + */ +@NonNullByDefault +@Component(configurationPid = "binding.haassohnpelletstove", service = ThingHandlerFactory.class) +public class HaasSohnpelletstoveHandlerFactory extends BaseThingHandlerFactory { + + private static final Set SUPPORTED_THING_TYPES_UIDS = Set.of(THING_TYPE_OVEN); + + @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_OVEN.equals(thingTypeUID)) { + return new HaasSohnpelletstoveHandler(thing); + } + return null; + } +} diff --git a/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveJSONCommunication.java b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveJSONCommunication.java new file mode 100644 index 000000000..04c205978 --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveJSONCommunication.java @@ -0,0 +1,223 @@ +/** + * Copyright (c) 2010-2021 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.haassohnpelletstove.internal; + +import java.io.ByteArrayInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.UnsupportedEncodingException; +import java.util.Properties; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.openhab.core.io.net.http.HttpUtil; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.google.gson.Gson; + +/** + * This class handles the JSON communication with the Wifi Modul of the Stove + * + * @author Christian Feininger - Initial contribution + * + */ +@NonNullByDefault +public class HaasSohnpelletstoveJSONCommunication { + + private final Logger logger = LoggerFactory.getLogger(HaasSohnpelletstoveJSONCommunication.class); + private HaasSohnpelletstoveConfiguration config; + + private Gson gson; + private @Nullable String xhspin; + private @Nullable HaasSohnpelletstoveJsonDataDTO ovenData; + + public HaasSohnpelletstoveJSONCommunication() { + gson = new Gson(); + ovenData = new HaasSohnpelletstoveJsonDataDTO(); + xhspin = ""; + config = new HaasSohnpelletstoveConfiguration(); + } + + /** + * Refreshes the oven Connection with the internal oven token. + * + * @param message Message object to pass errors to the calling method. + * @param thingUID Thing UID for logging purposes + * @return true if no error occurred, false otherwise. + */ + public boolean refreshOvenConnection(Helper message, String thingUID) { + if (config.hostIP == null || config.hostPIN == null) { + message.setStatusDescription("Error in configuration. Please recreate Thing."); + return false; + } + HaasSohnpelletstoveJsonDataDTO result = null; + boolean resultOk = false; + String error = "", errorDetail = "", statusDescr = ""; + String urlStr = "http://" + config.hostIP + "/status.cgi"; + + String response = null; + try { + response = HttpUtil.executeUrl("GET", urlStr, 10000); + logger.debug("OvenData = {}", response); + result = gson.fromJson(response, HaasSohnpelletstoveJsonDataDTO.class); + resultOk = true; + } catch (IOException e) { + logger.debug("Error processiong Get request {}", urlStr); + statusDescr = "Timeout error with" + config.hostIP + + ". Cannot find service on give IP. Please verify the IP-Address!"; + errorDetail = e.getMessage(); + resultOk = false; + } catch (Exception e) { + logger.debug("Unknwon Error: {}", e.getMessage()); + errorDetail = e.getMessage(); + resultOk = false; + } + if (resultOk) { + ovenData = result; + xhspin = getValidXHSPIN(ovenData); + } else { + logger.debug("Setting thing '{}' to OFFLINE: Error '{}': {}", thingUID, error, errorDetail); + ovenData = new HaasSohnpelletstoveJsonDataDTO(); + } + message.setStatusDescription(statusDescr); + return resultOk; + } + + /** + * Gets the status of the oven + * + * @return true if success or false in case of error + */ + public boolean updateOvenData(@Nullable String postData, Helper helper, String thingUID) { + String statusDescr = ""; + boolean resultOk = false; + String error = "", errorDetail = ""; + if (config.hostIP == null || config.hostPIN == null) { + return false; + } + String urlStr = "http://" + config.hostIP + "/status.cgi"; + + // Run the HTTP POST request and get the JSON response from Oven + String response = null; + + Properties httpHeader = new Properties(); + + if (postData != null) { + try { + InputStream targetStream = new ByteArrayInputStream(postData.getBytes("UTF-8")); + refreshOvenConnection(helper, thingUID); + httpHeader = createHeader(postData); + response = HttpUtil.executeUrl("POST", urlStr, httpHeader, targetStream, "application/json", 10000); + resultOk = true; + logger.debug("Execute POST request with content to {} with header: {}", urlStr, httpHeader.toString()); + } catch (UnsupportedEncodingException e1) { + logger.debug("Wrong encoding found. Only UTF-8 is supported."); + statusDescr = "Encoding of oven is not supported. Only UTF-8 is supported."; + resultOk = false; + } catch (IOException e) { + logger.debug("Error processiong POST request {}", urlStr); + statusDescr = "Cannot execute command on Stove. Please verify connection and Thing Status"; + resultOk = false; + } + } else { + try { + refreshOvenConnection(helper, thingUID); + httpHeader = createHeader(null); + response = HttpUtil.executeUrl("POST", urlStr, httpHeader, null, "", 10000); + resultOk = true; + logger.debug("Execute POST request to {} with header: {}", urlStr, httpHeader.toString()); + } catch (IOException e) { + logger.debug("Error processiong POST request {}", e.getMessage()); + String message = e.getMessage(); + if (message != null && message.contains("Authentication challenge without WWW-Authenticate ")) { + statusDescr = "Cannot connect to stove. Given PIN: " + config.hostPIN + " is incorrect!"; + } + resultOk = false; + } + } + if (resultOk) { + logger.debug("OvenData = {}", response); + ovenData = gson.fromJson(response, HaasSohnpelletstoveJsonDataDTO.class); + } else { + logger.debug("Setting thing '{}' to OFFLINE: Error '{}': {}", thingUID, error, errorDetail); + ovenData = new HaasSohnpelletstoveJsonDataDTO(); + } + helper.setStatusDescription(statusDescr); + return resultOk; + } + + /** + * Creates the header for the Post Request + * + * @return The created Header Properties + * @throws UnsupportedEncodingException + */ + private Properties createHeader(@Nullable String postData) throws UnsupportedEncodingException { + Properties httpHeader = new Properties(); + httpHeader.setProperty("Host", config.hostIP); + httpHeader.setProperty("Accept", "*/*"); + httpHeader.setProperty("Proxy-Connection", "keep-alive"); + httpHeader.setProperty("X-BACKEND-IP", "https://app.haassohn.com"); + httpHeader.setProperty("Accept-Language", "de-DE;q=1.0, en-DE;q=0.9"); + httpHeader.setProperty("Accept-Encoding", "gzip;q=1.0, compress;q=0.5"); + httpHeader.setProperty("token", "32 bytes"); + httpHeader.setProperty("Content-Type", "application/json"); + if (postData != null) { + int a = postData.getBytes("UTF-8").length; + httpHeader.setProperty(xhspin, Integer.toString(a)); + } + httpHeader.setProperty("User-Agent", "ios"); + httpHeader.setProperty("Connection", "keep-alive"); + httpHeader.setProperty("X-HS-PIN", xhspin); + return httpHeader; + } + + /** + * Generate the valid encrypted string to communicate with the oven. + * + * @param ovenData + * @return + */ + private @Nullable String getValidXHSPIN(@Nullable HaasSohnpelletstoveJsonDataDTO ovenData) { + if (ovenData != null && config.hostPIN != null) { + String nonce = ovenData.getNonce(); + String hostPIN = config.hostPIN; + String ePin = MD5Utils.getMD5String(hostPIN); + return MD5Utils.getMD5String(nonce + ePin); + } else { + return null; + } + } + + /** + * Set the config for service to communicate + * + * @param config2 + */ + public void setConfig(@Nullable HaasSohnpelletstoveConfiguration config2) { + if (config2 != null) { + this.config = config2; + } + } + + /** + * Returns the actual stored Oven Data + * + * @return + */ + @Nullable + public HaasSohnpelletstoveJsonDataDTO getOvenData() { + return this.ovenData; + } +} diff --git a/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveJsonDataDTO.java b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveJsonDataDTO.java new file mode 100644 index 000000000..17c45def8 --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/HaasSohnpelletstoveJsonDataDTO.java @@ -0,0 +1,152 @@ +/** + * Copyright (c) 2010-2021 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.haassohnpelletstove.internal; + +import com.google.gson.annotations.SerializedName; + +/** + * The {@link HaasSohnpelletstoveJsonDataDTO} is the Java class used to map the JSON + * response to a Oven request. + * + * @author Christian Feininger - Initial contribution + */ +public class HaasSohnpelletstoveJsonDataDTO { + metadata meta = new metadata(); + boolean prg; + boolean wprg; + String mode = ""; + @SerializedName("sp_temp") + String spTemp = ""; + @SerializedName("is_temp") + String isTemp = ""; + @SerializedName("ht_char") + String htChar = ""; + @SerializedName("weekprogram") + private wprogram[] weekprogram; + @SerializedName("error") + private err[] error; + @SerializedName("eco_mode") + boolean ecoMode; + boolean pgi; + String ignitions = ""; + @SerializedName("on_time") + String onTime = ""; + String consumption = ""; + @SerializedName("maintenance_in") + String maintenanceIn = ""; + @SerializedName("cleaning_in") + String cleaningIn = ""; + + /*** + * Get the nonce + * + * @return nonce + */ + public String getNonce() { + return this.meta.getNonce(); + } + + /** + * Returns the is Temperature of the Oven + * + * @return + */ + public String getisTemp() { + return isTemp; + } + + public boolean getEcoMode() { + return ecoMode; + } + + public String getIgnitions() { + return ignitions; + } + + public String getOnTime() { + return onTime; + } + + public String getConsumption() { + return consumption; + } + + public String getMaintenanceIn() { + return maintenanceIn; + } + + public String getCleaningIn() { + return cleaningIn; + } + + /*** + * JSON response + * + * @return JSON response as object + */ + public HaasSohnpelletstoveJsonDataDTO getResponse() { + return this; + } + + public class metadata { + @SerializedName("sw_version") + String swVersion = ""; + @SerializedName("hw_version") + String hwVersion = ""; + @SerializedName("bootl_version") + String bootlVersion = ""; + @SerializedName("wifi_sw_version") + String wifiSWVersion = ""; + @SerializedName("wifi_bootl_version") + String wifiBootlVersion = ""; + String sn = ""; + String typ = ""; + String language = ""; + String nonce = ""; + @SerializedName("eco_editable") + String ecoEditable = ""; + String ts = ""; + String ean = ""; + boolean rau; + @SerializedName("wlan_features") + private String[] wlan_features; + + public String getNonce() { + return nonce; + } + } + + public class err { + String time = ""; + String nr = ""; + } + + public class wprogram { + String day = ""; + String begin = ""; + String end = ""; + String temp = ""; + } + + public String getMode() { + return mode; + } + + public String getspTemp() { + return spTemp; + } + + public boolean getPrg() { + return prg; + } +} diff --git a/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/Helper.java b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/Helper.java new file mode 100644 index 000000000..5f279dc6c --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/Helper.java @@ -0,0 +1,48 @@ +/** + * Copyright (c) 2010-2021 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.haassohnpelletstove.internal; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; + +/** + * The {@link Helper} is a Helper class to overcome Call by value for a Status Description. + * + * + * @author Christian Feininger - Initial contribution + */ +@NonNullByDefault +public class Helper { + + private String statusDescription = ""; + + /*** + * Gets the Status Description + * + * @return + */ + public String getStatusDesription() { + return statusDescription; + } + + /*** + * Sets the Status Description + * + * @param status + */ + public void setStatusDescription(@Nullable String status) { + if (status != null) { + statusDescription = statusDescription + "\n" + status; + } + } +} diff --git a/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/MD5Utils.java b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/MD5Utils.java new file mode 100644 index 000000000..e1d1da009 --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/src/main/java/org/openhab/binding/haassohnpelletstove/internal/MD5Utils.java @@ -0,0 +1,66 @@ +/** + * Copyright (c) 2010-2021 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.haassohnpelletstove.internal; + +import java.nio.charset.Charset; +import java.nio.charset.StandardCharsets; +import java.security.MessageDigest; +import java.security.NoSuchAlgorithmException; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; + +/** + * The {@link MD5Utils} is responsible for generating the MD5 hash + * + * + * @author Christian Feininger - Initial contribution + */ +@NonNullByDefault +public class MD5Utils { + + private static final Charset UTF_8 = StandardCharsets.UTF_8; + + private static byte[] digest(byte[] input) { + MessageDigest md; + try { + md = MessageDigest.getInstance("MD5"); + } catch (NoSuchAlgorithmException e) { + throw new IllegalArgumentException(e); + } + byte[] result = md.digest(input); + return result; + } + + private static String bytesToHex(byte[] bytes) { + StringBuilder sb = new StringBuilder(); + for (byte b : bytes) { + sb.append(String.format("%02x", b)); + } + return sb.toString(); + } + + /*** + * Returns an encrypted MD5 string + * + * @param input nonce as input + * @return Encrypted String + */ + public static String getMD5String(@Nullable String input) { + if (input != null) { + byte[] md5InBytes = MD5Utils.digest(input.getBytes(UTF_8)); + return bytesToHex(md5InBytes); + } + return ""; + } +} diff --git a/bundles/org.openhab.binding.haassohnpelletstove/src/main/resources/OH-INF/binding/binding.xml b/bundles/org.openhab.binding.haassohnpelletstove/src/main/resources/OH-INF/binding/binding.xml new file mode 100644 index 000000000..8dd3fe21d --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/src/main/resources/OH-INF/binding/binding.xml @@ -0,0 +1,10 @@ + + + + Haas and Sohn Pelletstove Binding + This binding communicates with Haas and Sohn Pelletstoves through the optional WIFI module. It allows to + power the stove on and off and receives different operation information. + + diff --git a/bundles/org.openhab.binding.haassohnpelletstove/src/main/resources/OH-INF/thing/thing-types.xml b/bundles/org.openhab.binding.haassohnpelletstove/src/main/resources/OH-INF/thing/thing-types.xml new file mode 100644 index 000000000..dc0d1a74e --- /dev/null +++ b/bundles/org.openhab.binding.haassohnpelletstove/src/main/resources/OH-INF/thing/thing-types.xml @@ -0,0 +1,117 @@ + + + + + + + The binding for Haas and Sohn Pelletstove communicates with a Haas and Sohn Pelletstove through the + optional + WLAN-Modul. More information can be found here: https://www.haassohn.com/de/ihr-plus/WLAN-Funktion. It allows + to power on/off the stove as well as receiving different operation information about the stove. + + + + + + + + + + + + + + + + + + + Please add the IP Address of the WIFI Module of the Haas and Sohn oven here + network-address + + + + Please add the PIN of your oven here. You can find it in the Menu directly in your oven. + + + + How often the Pellet Stove should schedule a refresh after a channel is linked to an item. Temperature + data will be refreshed according this set time in seconds. Valid input is 0 - 999. + + true + 30 + + + + + + Number:Temperature + + Receives the is temperature of the stove as number:temperature + + + + + String + + Receives the actual mode of the stove as string + + + + + Number:Temperature + + Set the target temperature of the stove as number:temperature + + + + Switch + + To turn the stove on/off as switch + + + + Switch + + To turn the Eco Mode on/off for the stove as switch + + + + Number + + Receives the total amount of ignitions of the stove as string + + + + + Number:Mass + + Provides a pellet forecast when the stove need to be maintained next in kilogram as number:mass + + + + + String + + Provides a time forecast in hours:minutes when the stove need to be cleaned next as String + + + + + Number:Mass + + Provides the information about the total consumption of pellets of the stove as number:mass + + + + + Number + + Provides the information of the operating hours of stove as number + + + + diff --git a/bundles/pom.xml b/bundles/pom.xml index bf2fe504d..f3395fa01 100644 --- a/bundles/pom.xml +++ b/bundles/pom.xml @@ -129,6 +129,7 @@ org.openhab.binding.gpstracker org.openhab.binding.gree org.openhab.binding.groheondus + org.openhab.binding.haassohnpelletstove org.openhab.binding.harmonyhub org.openhab.binding.haywardomnilogic org.openhab.binding.hdanywhere