[mqtt.homeassistant] Implement Humidifier (#17853)

* [mqtt.homeassistant] Implement Humidifier

Signed-off-by: Cody Cutrer <cody@cutrer.us>

* remove debug logging from test

Signed-off-by: Cody Cutrer <cody@cutrer.us>
This commit is contained in:
Cody Cutrer 2024-12-06 16:39:43 -07:00 committed by GitHub
parent 2e7f0e061c
commit f733c85343
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
8 changed files with 328 additions and 2 deletions

View File

@ -109,6 +109,18 @@ If a device has multiple device triggers for the same subtype (the particular bu
| direction | String | R/W | `forward` or `backward` | | direction | String | R/W | `forward` or `backward` |
| json-attributes | String | RO | Additional attributes, as a serialized JSON string. | | json-attributes | String | RO | Additional attributes, as a serialized JSON string. |
### [Humidifier](https://www.home-assistant.io/integrations/humidifier.mqtt/)
| Channel ID | Type | R/W | Description |
|------------------|----------------------|-----|------------------------------------------------------------------------------------------|
| state | Switch | R/W | If the humidifier should be on or off. |
| action | String | RO | What the humidifier is actively doing. One of `off`, `humidifying`, `drying`, or `idle`. |
| mode | String | R/W | Inspect the state description for valid values. |
| current-humidity | Number:Dimensionless | RO | The current detected relative humidity, in %. |
| target-humidity | Number:Dimensionless | R/W | The desired relative humidity, in %. |
| device-class | String | RO | `humidifier` or `dehumidifier` |
| json-attributes | String | RO | Additional attributes, as a serialized JSON string. |
### [Light](https://www.home-assistant.io/integrations/light.mqtt/) ### [Light](https://www.home-assistant.io/integrations/light.mqtt/)
| Channel ID | Type | R/W | Description | | Channel ID | Type | R/W | Description |

View File

@ -32,6 +32,7 @@ public enum ComponentChannelType {
STRING("ha-string"), STRING("ha-string"),
SWITCH("ha-switch"), SWITCH("ha-switch"),
TRIGGER("ha-trigger"), TRIGGER("ha-trigger"),
HUMIDITY("ha-humidity"),
GPS_ACCURACY("ha-gps-accuracy"); GPS_ACCURACY("ha-gps-accuracy");
final ChannelTypeUID channelTypeUID; final ChannelTypeUID channelTypeUID;

View File

@ -272,7 +272,7 @@ public class Climate extends AbstractComponent<Climate.ChannelConfiguration> {
channelConfiguration.awayModeCommandTopic, channelConfiguration.awayModeStateTemplate, channelConfiguration.awayModeCommandTopic, channelConfiguration.awayModeStateTemplate,
channelConfiguration.awayModeStateTopic, commandFilter); channelConfiguration.awayModeStateTopic, commandFilter);
buildOptionalChannel(CURRENT_HUMIDITY_CH_ID, ComponentChannelType.NUMBER, buildOptionalChannel(CURRENT_HUMIDITY_CH_ID, ComponentChannelType.HUMIDITY,
new NumberValue(new BigDecimal(0), new BigDecimal(100), null, Units.PERCENT), updateListener, null, new NumberValue(new BigDecimal(0), new BigDecimal(100), null, Units.PERCENT), updateListener, null,
null, channelConfiguration.currentHumidityTemplate, channelConfiguration.currentHumidityTopic, null); null, channelConfiguration.currentHumidityTemplate, channelConfiguration.currentHumidityTopic, null);
@ -310,7 +310,7 @@ public class Climate extends AbstractComponent<Climate.ChannelConfiguration> {
channelConfiguration.swingCommandTemplate, channelConfiguration.swingCommandTopic, channelConfiguration.swingCommandTemplate, channelConfiguration.swingCommandTopic,
channelConfiguration.swingStateTemplate, channelConfiguration.swingStateTopic, commandFilter); channelConfiguration.swingStateTemplate, channelConfiguration.swingStateTopic, commandFilter);
buildOptionalChannel(TARGET_HUMIDITY_CH_ID, ComponentChannelType.NUMBER, buildOptionalChannel(TARGET_HUMIDITY_CH_ID, ComponentChannelType.HUMIDITY,
new NumberValue(channelConfiguration.minHumidity, channelConfiguration.maxHumidity, null, new NumberValue(channelConfiguration.minHumidity, channelConfiguration.maxHumidity, null,
Units.PERCENT), Units.PERCENT),
updateListener, channelConfiguration.targetHumidityCommandTemplate, updateListener, channelConfiguration.targetHumidityCommandTemplate,

View File

@ -71,6 +71,8 @@ public class ComponentFactory {
return new Event(componentConfiguration, newStyleChannels); return new Event(componentConfiguration, newStyleChannels);
case "fan": case "fan":
return new Fan(componentConfiguration, newStyleChannels); return new Fan(componentConfiguration, newStyleChannels);
case "humidifier":
return new Humidifier(componentConfiguration, newStyleChannels);
case "light": case "light":
return Light.create(componentConfiguration, newStyleChannels); return Light.create(componentConfiguration, newStyleChannels);
case "lock": case "lock":

View File

@ -0,0 +1,174 @@
/**
* 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.mqtt.homeassistant.internal.component;
import java.math.BigDecimal;
import java.util.List;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.openhab.binding.mqtt.generic.values.NumberValue;
import org.openhab.binding.mqtt.generic.values.OnOffValue;
import org.openhab.binding.mqtt.generic.values.TextValue;
import org.openhab.binding.mqtt.homeassistant.internal.ComponentChannelType;
import org.openhab.binding.mqtt.homeassistant.internal.config.dto.AbstractChannelConfiguration;
import org.openhab.binding.mqtt.homeassistant.internal.exception.ConfigurationException;
import org.openhab.core.library.types.StringType;
import org.openhab.core.library.unit.Units;
import com.google.gson.annotations.SerializedName;
/**
* A MQTT Humidifier, following the https://www.home-assistant.io/integrations/humidifier.mqtt/ specification.
*
* @author Cody Cutrer - Initial contribution
*/
@NonNullByDefault
public class Humidifier extends AbstractComponent<Humidifier.ChannelConfiguration> {
public static final String ACTION_CHANNEL_ID = "action";
public static final String CURRENT_HUMIDITY_CHANNEL_ID = "current-humidity";
public static final String DEVICE_CLASS_CHANNEL_ID = "device-class";
public static final String MODE_CHANNEL_ID = "mode";
public static final String STATE_CHANNEL_ID = "state";
public static final String TARGET_HUMIDITY_CHANNEL_ID = "target-humidity";
public static final String PLATFORM_HUMIDIFIER = "humidifier";
public static final String[] ACTIONS = new String[] { "off", "humidifying", "drying", "idle" };
/**
* Configuration class for MQTT component
*/
static class ChannelConfiguration extends AbstractChannelConfiguration {
ChannelConfiguration() {
super("MQTT Humidifier");
}
protected @Nullable Boolean optimistic;
@SerializedName("action_topic")
protected @Nullable String actionTopic;
@SerializedName("action_template")
protected @Nullable String actionTemplate;
@SerializedName("command_topic")
protected String commandTopic = "";
@SerializedName("command_template")
protected @Nullable String commandTemplate;
@SerializedName("state_topic")
protected @Nullable String stateTopic;
@SerializedName("state_value_template")
protected @Nullable String stateValueTemplate;
@SerializedName("current_humidity_topic")
protected @Nullable String currentHumidityTopic;
@SerializedName("current_humidity_template")
protected @Nullable String currentHumidityTemplate;
@SerializedName("target_humidity_command_topic")
protected @Nullable String targetHumidityCommandTopic;
@SerializedName("target_humidity_command_template")
protected @Nullable String targetHumidityCommandTemplate;
@SerializedName("target_humidity_state_topic")
protected @Nullable String targetHumidityStateTopic;
@SerializedName("target_humidity_state_template")
protected @Nullable String targetHumidityStateTemplate;
@SerializedName("mode_command_topic")
protected @Nullable String modeCommandTopic;
@SerializedName("mode_command_template")
protected @Nullable String modeCommandTemplate;
@SerializedName("mode_state_topic")
protected @Nullable String modeStateTopic;
@SerializedName("mode_state_template")
protected @Nullable String modeStateTemplate;
@SerializedName("device_class")
protected @Nullable String deviceClass;
protected String platform = "";
@SerializedName("min_humidity")
protected BigDecimal minHumidity = BigDecimal.ZERO;
@SerializedName("max_humidity")
protected BigDecimal maxHumidity = new BigDecimal(100);
@SerializedName("payload_on")
protected String payloadOn = "ON";
@SerializedName("payload_off")
protected String payloadOff = "OFF";
@SerializedName("payload_reset_humidity")
protected String payloadResetHumidity = "None";
@SerializedName("payload_reset_mode")
protected String payloadResetMode = "None";
protected @Nullable List<String> modes;
}
public Humidifier(ComponentFactory.ComponentConfiguration componentConfiguration, boolean newStyleChannels) {
super(componentConfiguration, ChannelConfiguration.class, newStyleChannels);
if (!PLATFORM_HUMIDIFIER.equals(channelConfiguration.platform)) {
throw new ConfigurationException("platform must be " + PLATFORM_HUMIDIFIER);
}
buildChannel(STATE_CHANNEL_ID, ComponentChannelType.SWITCH,
new OnOffValue(channelConfiguration.payloadOn, channelConfiguration.payloadOff), "State",
componentConfiguration.getUpdateListener())
.stateTopic(channelConfiguration.stateTopic, channelConfiguration.stateValueTemplate)
.commandTopic(channelConfiguration.commandTopic, channelConfiguration.isRetain(),
channelConfiguration.getQos(), channelConfiguration.commandTemplate)
.inferOptimistic(channelConfiguration.optimistic).build();
buildChannel(TARGET_HUMIDITY_CHANNEL_ID, ComponentChannelType.HUMIDITY,
new NumberValue(channelConfiguration.minHumidity, channelConfiguration.maxHumidity, null,
Units.PERCENT),
"Target Humidity", componentConfiguration.getUpdateListener())
.stateTopic(channelConfiguration.targetHumidityStateTopic,
channelConfiguration.targetHumidityStateTemplate)
.commandTopic(channelConfiguration.targetHumidityCommandTopic, channelConfiguration.isRetain(),
channelConfiguration.getQos(), channelConfiguration.targetHumidityCommandTemplate)
.inferOptimistic(channelConfiguration.optimistic).build();
if (channelConfiguration.actionTopic != null) {
buildChannel(ACTION_CHANNEL_ID, ComponentChannelType.STRING, new TextValue(ACTIONS), "Action",
componentConfiguration.getUpdateListener())
.stateTopic(channelConfiguration.actionTopic, channelConfiguration.actionTemplate).build();
}
if (channelConfiguration.modeCommandTopic != null) {
List<String> modes = channelConfiguration.modes;
if (modes == null) {
throw new ConfigurationException("modes cannot be null if mode_command_topic is specified");
}
TextValue modeValue = new TextValue(modes.toArray(new String[0]));
modeValue.setNullValue(channelConfiguration.payloadResetMode);
buildChannel(MODE_CHANNEL_ID, ComponentChannelType.STRING, modeValue, "Mode",
componentConfiguration.getUpdateListener())
.stateTopic(channelConfiguration.modeStateTopic, channelConfiguration.modeStateTemplate)
.commandTopic(channelConfiguration.modeCommandTopic, channelConfiguration.isRetain(),
channelConfiguration.getQos(), channelConfiguration.modeCommandTemplate)
.inferOptimistic(channelConfiguration.optimistic).build();
}
if (channelConfiguration.currentHumidityTopic != null) {
buildChannel(CURRENT_HUMIDITY_CHANNEL_ID, ComponentChannelType.HUMIDITY,
new NumberValue(null, null, null, Units.PERCENT), "Current Humidity",
componentConfiguration.getUpdateListener())
.stateTopic(channelConfiguration.currentHumidityTopic, channelConfiguration.currentHumidityTemplate)
.build();
}
if (channelConfiguration.deviceClass != null) {
TextValue deviceClassValue = new TextValue();
deviceClassValue.update(new StringType(channelConfiguration.deviceClass));
buildChannel(DEVICE_CLASS_CHANNEL_ID, ComponentChannelType.STRING, deviceClassValue, "Device Class",
componentConfiguration.getUpdateListener()).build();
}
finalizeChannels();
}
}

View File

@ -18,6 +18,7 @@ channel-type.mqtt.ha-dimmer-advanced.label = Dimmer
channel-type.mqtt.ha-dimmer.label = Dimmer channel-type.mqtt.ha-dimmer.label = Dimmer
channel-type.mqtt.ha-gps-accuracy.label = GPS Accuracy channel-type.mqtt.ha-gps-accuracy.label = GPS Accuracy
channel-type.mqtt.ha-gps-accuracy.description = The accuracy of the GPS fix, in meters. channel-type.mqtt.ha-gps-accuracy.description = The accuracy of the GPS fix, in meters.
channel-type.mqtt.ha-humidity.label = Humidity
channel-type.mqtt.ha-image-advanced.label = Image channel-type.mqtt.ha-image-advanced.label = Image
channel-type.mqtt.ha-image.label = Image channel-type.mqtt.ha-image.label = Image
channel-type.mqtt.ha-location.label = Location channel-type.mqtt.ha-location.label = Location

View File

@ -16,6 +16,12 @@
<config-description-ref uri="channel-type:mqtt:ha-channel"/> <config-description-ref uri="channel-type:mqtt:ha-channel"/>
</channel-type> </channel-type>
<channel-type id="ha-humidity">
<item-type unitHint="%">Number:Dimensionless</item-type>
<label>Humidity</label>
<config-description-ref uri="channel-type:mqtt:ha-channel"/>
</channel-type>
<channel-type id="ha-image"> <channel-type id="ha-image">
<item-type>Image</item-type> <item-type>Image</item-type>
<label>Image</label> <label>Image</label>

View File

@ -0,0 +1,130 @@
/**
* 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.mqtt.homeassistant.internal.component;
import static org.hamcrest.CoreMatchers.is;
import static org.hamcrest.MatcherAssert.assertThat;
import java.util.Set;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.junit.jupiter.api.Test;
import org.openhab.binding.mqtt.generic.values.NumberValue;
import org.openhab.binding.mqtt.generic.values.OnOffValue;
import org.openhab.binding.mqtt.generic.values.TextValue;
import org.openhab.core.library.types.DecimalType;
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.types.UnDefType;
/**
* Tests for {@link Humidifier}
*
* @author Cody Cutrer - Initial contribution
*/
@NonNullByDefault
public class HumidifierTests extends AbstractComponentTests {
public static final String CONFIG_TOPIC = "humidifier/bedroom_humidifier";
@SuppressWarnings("null")
@Test
public void test() {
var component = discoverComponent(configTopicToMqtt(CONFIG_TOPIC), """
{
"platform": "humidifier",
"name": "Bedroom humidifier",
"device_class": "humidifier",
"state_topic": "bedroom_humidifier/on/state",
"action_topic": "bedroom_humidifier/action",
"command_topic": "bedroom_humidifier/on/set",
"current_humidity_topic": "bedroom_humidifier/humidity/current",
"target_humidity_command_topic": "bedroom_humidifier/humidity/set",
"target_humidity_state_topic": "bedroom_humidifier/humidity/state",
"mode_state_topic": "bedroom_humidifier/mode/state",
"mode_command_topic": "bedroom_humidifier/preset/preset_mode",
"modes": [
"normal",
"eco",
"away",
"boost",
"comfort",
"home",
"sleep",
"auto",
"baby"],
"qos": 0,
"payload_on": "true",
"payload_off": "false",
"min_humidity": 30,
"max_humidity": 80
}
""");
assertThat(component.channels.size(), is(6));
assertThat(component.getName(), is("Bedroom humidifier"));
assertChannel(component, Humidifier.STATE_CHANNEL_ID, "bedroom_humidifier/on/state",
"bedroom_humidifier/on/set", "State", OnOffValue.class);
assertChannel(component, Humidifier.ACTION_CHANNEL_ID, "bedroom_humidifier/action", "", "Action",
TextValue.class);
assertChannel(component, Humidifier.MODE_CHANNEL_ID, "bedroom_humidifier/mode/state",
"bedroom_humidifier/preset/preset_mode", "Mode", TextValue.class);
assertChannel(component, Humidifier.DEVICE_CLASS_CHANNEL_ID, "", "", "Device Class", TextValue.class);
assertChannel(component, Humidifier.CURRENT_HUMIDITY_CHANNEL_ID, "bedroom_humidifier/humidity/current", "",
"Current Humidity", NumberValue.class);
assertChannel(component, Humidifier.TARGET_HUMIDITY_CHANNEL_ID, "bedroom_humidifier/humidity/state",
"bedroom_humidifier/humidity/set", "Target Humidity", NumberValue.class);
publishMessage("bedroom_humidifier/on/state", "true");
assertState(component, Humidifier.STATE_CHANNEL_ID, OnOffType.ON);
publishMessage("bedroom_humidifier/on/state", "false");
assertState(component, Humidifier.STATE_CHANNEL_ID, OnOffType.OFF);
publishMessage("bedroom_humidifier/action", "off");
assertState(component, Humidifier.ACTION_CHANNEL_ID, new StringType("off"));
publishMessage("bedroom_humidifier/action", "idle");
assertState(component, Humidifier.ACTION_CHANNEL_ID, new StringType("idle"));
publishMessage("bedroom_humidifier/action", "invalid");
assertState(component, Humidifier.ACTION_CHANNEL_ID, new StringType("idle"));
publishMessage("bedroom_humidifier/mode/state", "eco");
assertState(component, Humidifier.MODE_CHANNEL_ID, new StringType("eco"));
publishMessage("bedroom_humidifier/mode/state", "invalid");
assertState(component, Humidifier.MODE_CHANNEL_ID, new StringType("eco"));
publishMessage("bedroom_humidifier/mode/state", "None");
assertState(component, Humidifier.MODE_CHANNEL_ID, UnDefType.NULL);
publishMessage("bedroom_humidifier/humidity/current", "35");
assertState(component, Humidifier.CURRENT_HUMIDITY_CHANNEL_ID, new QuantityType<>(35, Units.PERCENT));
publishMessage("bedroom_humidifier/humidity/state", "40");
assertState(component, Humidifier.TARGET_HUMIDITY_CHANNEL_ID, new QuantityType<>(40, Units.PERCENT));
component.getChannel(Humidifier.STATE_CHANNEL_ID).getState().publishValue(OnOffType.OFF);
assertPublished("bedroom_humidifier/on/set", "false");
component.getChannel(Humidifier.STATE_CHANNEL_ID).getState().publishValue(OnOffType.ON);
assertPublished("bedroom_humidifier/on/set", "true");
component.getChannel(Humidifier.MODE_CHANNEL_ID).getState().publishValue(new StringType("eco"));
assertPublished("bedroom_humidifier/preset/preset_mode", "eco");
component.getChannel(Humidifier.TARGET_HUMIDITY_CHANNEL_ID).getState().publishValue(new DecimalType(45));
assertPublished("bedroom_humidifier/humidity/set", "45");
}
@Override
protected Set<String> getConfigTopics() {
return Set.of(CONFIG_TOPIC);
}
}