2019-04-22 19:49:15 +00:00
|
|
|
"""Support for Legacy MQTT vacuum."""
|
2019-04-14 18:09:46 +00:00
|
|
|
import json
|
2017-09-15 13:39:20 +00:00
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.components.vacuum import (
|
2021-06-29 09:21:22 +00:00
|
|
|
ATTR_STATUS,
|
2021-11-08 13:02:18 +00:00
|
|
|
ENTITY_ID_FORMAT,
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_BATTERY,
|
|
|
|
SUPPORT_CLEAN_SPOT,
|
|
|
|
SUPPORT_FAN_SPEED,
|
|
|
|
SUPPORT_LOCATE,
|
|
|
|
SUPPORT_PAUSE,
|
|
|
|
SUPPORT_RETURN_HOME,
|
|
|
|
SUPPORT_SEND_COMMAND,
|
|
|
|
SUPPORT_STATUS,
|
|
|
|
SUPPORT_STOP,
|
|
|
|
SUPPORT_TURN_OFF,
|
|
|
|
SUPPORT_TURN_ON,
|
2020-04-26 00:11:08 +00:00
|
|
|
VacuumEntity,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2021-03-11 12:42:13 +00:00
|
|
|
from homeassistant.const import ATTR_SUPPORTED_FEATURES, CONF_NAME
|
2017-09-15 13:39:20 +00:00
|
|
|
from homeassistant.core import callback
|
2018-01-21 06:35:38 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2017-10-06 03:55:19 +00:00
|
|
|
from homeassistant.helpers.icon import icon_for_battery_level
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2022-01-03 15:07:40 +00:00
|
|
|
from .. import MqttValueTemplate, subscription
|
2021-01-08 23:47:17 +00:00
|
|
|
from ... import mqtt
|
2022-01-03 08:03:47 +00:00
|
|
|
from ..const import CONF_COMMAND_TOPIC, CONF_ENCODING, CONF_QOS, CONF_RETAIN
|
2021-01-08 23:47:17 +00:00
|
|
|
from ..debug_info import log_messages
|
2021-03-11 12:42:13 +00:00
|
|
|
from ..mixins import MQTT_ENTITY_COMMON_SCHEMA, MqttEntity
|
2021-06-29 09:21:22 +00:00
|
|
|
from .const import MQTT_VACUUM_ATTRIBUTES_BLOCKED
|
2019-10-18 00:04:27 +00:00
|
|
|
from .schema import MQTT_VACUUM_SCHEMA, services_to_strings, strings_to_services
|
2019-03-21 05:56:46 +00:00
|
|
|
|
2017-09-15 13:39:20 +00:00
|
|
|
SERVICE_TO_STRING = {
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_TURN_ON: "turn_on",
|
|
|
|
SUPPORT_TURN_OFF: "turn_off",
|
|
|
|
SUPPORT_PAUSE: "pause",
|
|
|
|
SUPPORT_STOP: "stop",
|
|
|
|
SUPPORT_RETURN_HOME: "return_home",
|
|
|
|
SUPPORT_FAN_SPEED: "fan_speed",
|
|
|
|
SUPPORT_BATTERY: "battery",
|
|
|
|
SUPPORT_STATUS: "status",
|
|
|
|
SUPPORT_SEND_COMMAND: "send_command",
|
|
|
|
SUPPORT_LOCATE: "locate",
|
|
|
|
SUPPORT_CLEAN_SPOT: "clean_spot",
|
2017-09-15 13:39:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
STRING_TO_SERVICE = {v: k for k, v in SERVICE_TO_STRING.items()}
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_SERVICES = (
|
|
|
|
SUPPORT_TURN_ON
|
|
|
|
| SUPPORT_TURN_OFF
|
|
|
|
| SUPPORT_STOP
|
|
|
|
| SUPPORT_RETURN_HOME
|
|
|
|
| SUPPORT_STATUS
|
|
|
|
| SUPPORT_BATTERY
|
|
|
|
| SUPPORT_CLEAN_SPOT
|
|
|
|
)
|
|
|
|
ALL_SERVICES = (
|
|
|
|
DEFAULT_SERVICES
|
|
|
|
| SUPPORT_PAUSE
|
|
|
|
| SUPPORT_LOCATE
|
|
|
|
| SUPPORT_FAN_SPEED
|
|
|
|
| SUPPORT_SEND_COMMAND
|
|
|
|
)
|
2017-09-15 13:39:20 +00:00
|
|
|
|
|
|
|
CONF_SUPPORTED_FEATURES = ATTR_SUPPORTED_FEATURES
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_BATTERY_LEVEL_TEMPLATE = "battery_level_template"
|
|
|
|
CONF_BATTERY_LEVEL_TOPIC = "battery_level_topic"
|
|
|
|
CONF_CHARGING_TEMPLATE = "charging_template"
|
|
|
|
CONF_CHARGING_TOPIC = "charging_topic"
|
|
|
|
CONF_CLEANING_TEMPLATE = "cleaning_template"
|
|
|
|
CONF_CLEANING_TOPIC = "cleaning_topic"
|
|
|
|
CONF_DOCKED_TEMPLATE = "docked_template"
|
|
|
|
CONF_DOCKED_TOPIC = "docked_topic"
|
|
|
|
CONF_ERROR_TEMPLATE = "error_template"
|
|
|
|
CONF_ERROR_TOPIC = "error_topic"
|
|
|
|
CONF_FAN_SPEED_LIST = "fan_speed_list"
|
|
|
|
CONF_FAN_SPEED_TEMPLATE = "fan_speed_template"
|
|
|
|
CONF_FAN_SPEED_TOPIC = "fan_speed_topic"
|
|
|
|
CONF_PAYLOAD_CLEAN_SPOT = "payload_clean_spot"
|
|
|
|
CONF_PAYLOAD_LOCATE = "payload_locate"
|
|
|
|
CONF_PAYLOAD_RETURN_TO_BASE = "payload_return_to_base"
|
|
|
|
CONF_PAYLOAD_START_PAUSE = "payload_start_pause"
|
|
|
|
CONF_PAYLOAD_STOP = "payload_stop"
|
|
|
|
CONF_PAYLOAD_TURN_OFF = "payload_turn_off"
|
|
|
|
CONF_PAYLOAD_TURN_ON = "payload_turn_on"
|
|
|
|
CONF_SEND_COMMAND_TOPIC = "send_command_topic"
|
|
|
|
CONF_SET_FAN_SPEED_TOPIC = "set_fan_speed_topic"
|
|
|
|
|
|
|
|
DEFAULT_NAME = "MQTT Vacuum"
|
|
|
|
DEFAULT_PAYLOAD_CLEAN_SPOT = "clean_spot"
|
|
|
|
DEFAULT_PAYLOAD_LOCATE = "locate"
|
|
|
|
DEFAULT_PAYLOAD_RETURN_TO_BASE = "return_to_base"
|
|
|
|
DEFAULT_PAYLOAD_START_PAUSE = "start_pause"
|
|
|
|
DEFAULT_PAYLOAD_STOP = "stop"
|
|
|
|
DEFAULT_PAYLOAD_TURN_OFF = "turn_off"
|
|
|
|
DEFAULT_PAYLOAD_TURN_ON = "turn_on"
|
2019-04-14 03:29:01 +00:00
|
|
|
DEFAULT_RETAIN = False
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_SERVICE_STRINGS = services_to_strings(DEFAULT_SERVICES, SERVICE_TO_STRING)
|
|
|
|
|
2021-06-29 09:21:22 +00:00
|
|
|
MQTT_LEGACY_VACUUM_ATTRIBUTES_BLOCKED = MQTT_VACUUM_ATTRIBUTES_BLOCKED | frozenset(
|
|
|
|
{ATTR_STATUS}
|
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA_LEGACY = (
|
|
|
|
mqtt.MQTT_BASE_PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Inclusive(CONF_BATTERY_LEVEL_TEMPLATE, "battery"): cv.template,
|
|
|
|
vol.Inclusive(
|
|
|
|
CONF_BATTERY_LEVEL_TOPIC, "battery"
|
|
|
|
): mqtt.valid_publish_topic,
|
|
|
|
vol.Inclusive(CONF_CHARGING_TEMPLATE, "charging"): cv.template,
|
|
|
|
vol.Inclusive(CONF_CHARGING_TOPIC, "charging"): mqtt.valid_publish_topic,
|
|
|
|
vol.Inclusive(CONF_CLEANING_TEMPLATE, "cleaning"): cv.template,
|
|
|
|
vol.Inclusive(CONF_CLEANING_TOPIC, "cleaning"): mqtt.valid_publish_topic,
|
|
|
|
vol.Inclusive(CONF_DOCKED_TEMPLATE, "docked"): cv.template,
|
|
|
|
vol.Inclusive(CONF_DOCKED_TOPIC, "docked"): mqtt.valid_publish_topic,
|
|
|
|
vol.Inclusive(CONF_ERROR_TEMPLATE, "error"): cv.template,
|
|
|
|
vol.Inclusive(CONF_ERROR_TOPIC, "error"): mqtt.valid_publish_topic,
|
|
|
|
vol.Optional(CONF_FAN_SPEED_LIST, default=[]): vol.All(
|
|
|
|
cv.ensure_list, [cv.string]
|
|
|
|
),
|
|
|
|
vol.Inclusive(CONF_FAN_SPEED_TEMPLATE, "fan_speed"): cv.template,
|
|
|
|
vol.Inclusive(CONF_FAN_SPEED_TOPIC, "fan_speed"): mqtt.valid_publish_topic,
|
|
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
|
|
vol.Optional(
|
|
|
|
CONF_PAYLOAD_CLEAN_SPOT, default=DEFAULT_PAYLOAD_CLEAN_SPOT
|
|
|
|
): cv.string,
|
|
|
|
vol.Optional(
|
|
|
|
CONF_PAYLOAD_LOCATE, default=DEFAULT_PAYLOAD_LOCATE
|
|
|
|
): cv.string,
|
|
|
|
vol.Optional(
|
|
|
|
CONF_PAYLOAD_RETURN_TO_BASE, default=DEFAULT_PAYLOAD_RETURN_TO_BASE
|
|
|
|
): cv.string,
|
|
|
|
vol.Optional(
|
|
|
|
CONF_PAYLOAD_START_PAUSE, default=DEFAULT_PAYLOAD_START_PAUSE
|
|
|
|
): cv.string,
|
|
|
|
vol.Optional(CONF_PAYLOAD_STOP, default=DEFAULT_PAYLOAD_STOP): cv.string,
|
|
|
|
vol.Optional(
|
|
|
|
CONF_PAYLOAD_TURN_OFF, default=DEFAULT_PAYLOAD_TURN_OFF
|
|
|
|
): cv.string,
|
|
|
|
vol.Optional(
|
|
|
|
CONF_PAYLOAD_TURN_ON, default=DEFAULT_PAYLOAD_TURN_ON
|
|
|
|
): cv.string,
|
|
|
|
vol.Optional(CONF_SEND_COMMAND_TOPIC): mqtt.valid_publish_topic,
|
|
|
|
vol.Optional(CONF_SET_FAN_SPEED_TOPIC): mqtt.valid_publish_topic,
|
|
|
|
vol.Optional(
|
|
|
|
CONF_SUPPORTED_FEATURES, default=DEFAULT_SERVICE_STRINGS
|
|
|
|
): vol.All(cv.ensure_list, [vol.In(STRING_TO_SERVICE.keys())]),
|
2021-10-12 09:26:38 +00:00
|
|
|
vol.Optional(CONF_COMMAND_TOPIC): mqtt.valid_publish_topic,
|
|
|
|
vol.Optional(CONF_RETAIN, default=DEFAULT_RETAIN): cv.boolean,
|
2019-07-31 19:25:30 +00:00
|
|
|
}
|
|
|
|
)
|
2021-03-11 12:42:13 +00:00
|
|
|
.extend(MQTT_ENTITY_COMMON_SCHEMA.schema)
|
2019-07-31 19:25:30 +00:00
|
|
|
.extend(MQTT_VACUUM_SCHEMA.schema)
|
|
|
|
)
|
|
|
|
|
2021-10-25 11:47:06 +00:00
|
|
|
DISCOVERY_SCHEMA_LEGACY = PLATFORM_SCHEMA_LEGACY.extend({}, extra=vol.REMOVE_EXTRA)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
|
|
|
|
async def async_setup_entity_legacy(
|
2021-06-23 13:44:18 +00:00
|
|
|
hass, config, async_add_entities, config_entry, discovery_data
|
2019-07-31 19:25:30 +00:00
|
|
|
):
|
2019-04-22 19:49:15 +00:00
|
|
|
"""Set up a MQTT Vacuum Legacy."""
|
2021-06-23 13:44:18 +00:00
|
|
|
async_add_entities([MqttVacuum(hass, config, config_entry, discovery_data)])
|
2019-01-06 16:05:04 +00:00
|
|
|
|
|
|
|
|
2021-01-09 16:46:53 +00:00
|
|
|
class MqttVacuum(MqttEntity, VacuumEntity):
|
2019-04-22 19:49:15 +00:00
|
|
|
"""Representation of a MQTT-controlled legacy vacuum."""
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2021-11-08 13:02:18 +00:00
|
|
|
_entity_id_format = ENTITY_ID_FORMAT
|
2021-06-29 09:21:22 +00:00
|
|
|
_attributes_extra_blocked = MQTT_LEGACY_VACUUM_ATTRIBUTES_BLOCKED
|
|
|
|
|
2021-06-23 13:44:18 +00:00
|
|
|
def __init__(self, hass, config, config_entry, discovery_data):
|
2017-09-15 13:39:20 +00:00
|
|
|
"""Initialize the vacuum."""
|
|
|
|
self._cleaning = False
|
|
|
|
self._charging = False
|
|
|
|
self._docked = False
|
2018-10-26 09:07:39 +00:00
|
|
|
self._error = None
|
2019-07-31 19:25:30 +00:00
|
|
|
self._status = "Unknown"
|
2017-09-15 13:39:20 +00:00
|
|
|
self._battery_level = 0
|
2019-07-31 19:25:30 +00:00
|
|
|
self._fan_speed = "unknown"
|
2019-01-06 13:16:46 +00:00
|
|
|
self._fan_speed_list = []
|
|
|
|
|
2021-06-23 13:44:18 +00:00
|
|
|
MqttEntity.__init__(self, hass, config, config_entry, discovery_data)
|
2019-01-06 13:16:46 +00:00
|
|
|
|
2021-01-09 16:46:53 +00:00
|
|
|
@staticmethod
|
|
|
|
def config_schema():
|
|
|
|
"""Return the config schema."""
|
2021-10-25 11:47:06 +00:00
|
|
|
return DISCOVERY_SCHEMA_LEGACY
|
2019-01-06 13:16:46 +00:00
|
|
|
|
|
|
|
def _setup_from_config(self, config):
|
2019-04-07 14:10:57 +00:00
|
|
|
supported_feature_strings = config[CONF_SUPPORTED_FEATURES]
|
2019-01-06 13:16:46 +00:00
|
|
|
self._supported_features = strings_to_services(
|
2019-04-22 19:49:15 +00:00
|
|
|
supported_feature_strings, STRING_TO_SERVICE
|
2019-01-06 13:16:46 +00:00
|
|
|
)
|
2019-04-07 14:10:57 +00:00
|
|
|
self._fan_speed_list = config[CONF_FAN_SPEED_LIST]
|
2021-10-12 09:26:38 +00:00
|
|
|
self._qos = config[CONF_QOS]
|
|
|
|
self._retain = config[CONF_RETAIN]
|
2022-01-03 15:08:07 +00:00
|
|
|
self._encoding = config[CONF_ENCODING] or None
|
2019-01-06 13:16:46 +00:00
|
|
|
|
2021-10-12 09:26:38 +00:00
|
|
|
self._command_topic = config.get(CONF_COMMAND_TOPIC)
|
2019-01-06 13:16:46 +00:00
|
|
|
self._set_fan_speed_topic = config.get(CONF_SET_FAN_SPEED_TOPIC)
|
|
|
|
self._send_command_topic = config.get(CONF_SEND_COMMAND_TOPIC)
|
|
|
|
|
|
|
|
self._payloads = {
|
2019-07-31 19:25:30 +00:00
|
|
|
key: config.get(key)
|
|
|
|
for key in (
|
2019-01-06 13:16:46 +00:00
|
|
|
CONF_PAYLOAD_TURN_ON,
|
|
|
|
CONF_PAYLOAD_TURN_OFF,
|
|
|
|
CONF_PAYLOAD_RETURN_TO_BASE,
|
|
|
|
CONF_PAYLOAD_STOP,
|
|
|
|
CONF_PAYLOAD_CLEAN_SPOT,
|
|
|
|
CONF_PAYLOAD_LOCATE,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_PAYLOAD_START_PAUSE,
|
2019-01-06 13:16:46 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
self._state_topics = {
|
2019-07-31 19:25:30 +00:00
|
|
|
key: config.get(key)
|
|
|
|
for key in (
|
2019-01-06 13:16:46 +00:00
|
|
|
CONF_BATTERY_LEVEL_TOPIC,
|
|
|
|
CONF_CHARGING_TOPIC,
|
|
|
|
CONF_CLEANING_TOPIC,
|
|
|
|
CONF_DOCKED_TOPIC,
|
|
|
|
CONF_ERROR_TOPIC,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_FAN_SPEED_TOPIC,
|
2019-01-06 13:16:46 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
self._templates = {
|
2019-07-31 19:25:30 +00:00
|
|
|
key: config.get(key)
|
|
|
|
for key in (
|
2019-01-06 13:16:46 +00:00
|
|
|
CONF_BATTERY_LEVEL_TEMPLATE,
|
|
|
|
CONF_CHARGING_TEMPLATE,
|
|
|
|
CONF_CLEANING_TEMPLATE,
|
|
|
|
CONF_DOCKED_TEMPLATE,
|
|
|
|
CONF_ERROR_TEMPLATE,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_FAN_SPEED_TEMPLATE,
|
2019-01-06 13:16:46 +00:00
|
|
|
)
|
|
|
|
}
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2019-01-06 13:16:46 +00:00
|
|
|
async def _subscribe_topics(self):
|
|
|
|
"""(Re)Subscribe to topics."""
|
|
|
|
for tpl in self._templates.values():
|
|
|
|
if tpl is not None:
|
2022-01-03 15:07:40 +00:00
|
|
|
tpl = MqttValueTemplate(tpl, entity=self)
|
2017-09-15 13:39:20 +00:00
|
|
|
|
|
|
|
@callback
|
2020-04-01 18:48:32 +00:00
|
|
|
@log_messages(self.hass, self.entity_id)
|
2019-03-13 19:58:20 +00:00
|
|
|
def message_received(msg):
|
2017-09-15 13:39:20 +00:00
|
|
|
"""Handle new MQTT message."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if (
|
|
|
|
msg.topic == self._state_topics[CONF_BATTERY_LEVEL_TOPIC]
|
|
|
|
and self._templates[CONF_BATTERY_LEVEL_TEMPLATE]
|
|
|
|
):
|
|
|
|
battery_level = self._templates[
|
|
|
|
CONF_BATTERY_LEVEL_TEMPLATE
|
2022-01-03 15:07:40 +00:00
|
|
|
].async_render_with_possible_json_value(msg.payload, None)
|
2019-04-14 03:29:01 +00:00
|
|
|
if battery_level:
|
2017-09-15 13:39:20 +00:00
|
|
|
self._battery_level = int(battery_level)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if (
|
|
|
|
msg.topic == self._state_topics[CONF_CHARGING_TOPIC]
|
|
|
|
and self._templates[CONF_CHARGING_TEMPLATE]
|
|
|
|
):
|
|
|
|
charging = self._templates[
|
|
|
|
CONF_CHARGING_TEMPLATE
|
2022-01-03 15:07:40 +00:00
|
|
|
].async_render_with_possible_json_value(msg.payload, None)
|
2019-04-14 03:29:01 +00:00
|
|
|
if charging:
|
2017-09-15 16:50:22 +00:00
|
|
|
self._charging = cv.boolean(charging)
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if (
|
|
|
|
msg.topic == self._state_topics[CONF_CLEANING_TOPIC]
|
|
|
|
and self._templates[CONF_CLEANING_TEMPLATE]
|
|
|
|
):
|
|
|
|
cleaning = self._templates[
|
|
|
|
CONF_CLEANING_TEMPLATE
|
2022-01-03 15:07:40 +00:00
|
|
|
].async_render_with_possible_json_value(msg.payload, None)
|
2019-04-14 03:29:01 +00:00
|
|
|
if cleaning:
|
2017-09-15 16:50:22 +00:00
|
|
|
self._cleaning = cv.boolean(cleaning)
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if (
|
|
|
|
msg.topic == self._state_topics[CONF_DOCKED_TOPIC]
|
|
|
|
and self._templates[CONF_DOCKED_TEMPLATE]
|
|
|
|
):
|
|
|
|
docked = self._templates[
|
|
|
|
CONF_DOCKED_TEMPLATE
|
2022-01-03 15:07:40 +00:00
|
|
|
].async_render_with_possible_json_value(msg.payload, None)
|
2019-04-14 03:29:01 +00:00
|
|
|
if docked:
|
2017-09-15 16:50:22 +00:00
|
|
|
self._docked = cv.boolean(docked)
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if (
|
|
|
|
msg.topic == self._state_topics[CONF_ERROR_TOPIC]
|
|
|
|
and self._templates[CONF_ERROR_TEMPLATE]
|
|
|
|
):
|
|
|
|
error = self._templates[
|
|
|
|
CONF_ERROR_TEMPLATE
|
2022-01-03 15:07:40 +00:00
|
|
|
].async_render_with_possible_json_value(msg.payload, None)
|
2019-04-19 04:01:19 +00:00
|
|
|
if error is not None:
|
2018-10-26 09:07:39 +00:00
|
|
|
self._error = cv.string(error)
|
|
|
|
|
2017-09-15 13:39:20 +00:00
|
|
|
if self._docked:
|
|
|
|
if self._charging:
|
|
|
|
self._status = "Docked & Charging"
|
|
|
|
else:
|
|
|
|
self._status = "Docked"
|
|
|
|
elif self._cleaning:
|
|
|
|
self._status = "Cleaning"
|
2019-04-14 03:29:01 +00:00
|
|
|
elif self._error:
|
2019-09-03 19:14:00 +00:00
|
|
|
self._status = f"Error: {self._error}"
|
2017-09-15 13:39:20 +00:00
|
|
|
else:
|
|
|
|
self._status = "Stopped"
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if (
|
|
|
|
msg.topic == self._state_topics[CONF_FAN_SPEED_TOPIC]
|
|
|
|
and self._templates[CONF_FAN_SPEED_TEMPLATE]
|
|
|
|
):
|
|
|
|
fan_speed = self._templates[
|
|
|
|
CONF_FAN_SPEED_TEMPLATE
|
2022-01-03 15:07:40 +00:00
|
|
|
].async_render_with_possible_json_value(msg.payload, None)
|
2019-04-14 03:29:01 +00:00
|
|
|
if fan_speed:
|
2017-09-15 13:39:20 +00:00
|
|
|
self._fan_speed = fan_speed
|
|
|
|
|
2019-03-12 21:46:48 +00:00
|
|
|
self.async_write_ha_state()
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2019-01-06 13:16:46 +00:00
|
|
|
topics_list = {topic for topic in self._state_topics.values() if topic}
|
|
|
|
self._sub_state = await subscription.async_subscribe_topics(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass,
|
|
|
|
self._sub_state,
|
2019-01-06 13:16:46 +00:00
|
|
|
{
|
2019-09-03 19:14:00 +00:00
|
|
|
f"topic{i}": {
|
2019-01-06 13:16:46 +00:00
|
|
|
"topic": topic,
|
|
|
|
"msg_callback": message_received,
|
2019-07-31 19:25:30 +00:00
|
|
|
"qos": self._qos,
|
2022-01-03 15:08:07 +00:00
|
|
|
"encoding": self._encoding,
|
2019-07-31 19:25:30 +00:00
|
|
|
}
|
|
|
|
for i, topic in enumerate(topics_list)
|
|
|
|
},
|
2019-01-06 13:16:46 +00:00
|
|
|
)
|
2017-09-15 13:39:20 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
|
|
|
"""Return true if vacuum is on."""
|
|
|
|
return self._cleaning
|
|
|
|
|
|
|
|
@property
|
|
|
|
def status(self):
|
|
|
|
"""Return a status string for the vacuum."""
|
|
|
|
return self._status
|
|
|
|
|
|
|
|
@property
|
|
|
|
def fan_speed(self):
|
|
|
|
"""Return the status of the vacuum."""
|
|
|
|
return self._fan_speed
|
|
|
|
|
|
|
|
@property
|
|
|
|
def fan_speed_list(self):
|
2020-05-26 00:56:12 +00:00
|
|
|
"""Return the status of the vacuum."""
|
2017-09-15 13:39:20 +00:00
|
|
|
return self._fan_speed_list
|
|
|
|
|
|
|
|
@property
|
|
|
|
def battery_level(self):
|
|
|
|
"""Return the status of the vacuum."""
|
|
|
|
return max(0, min(100, self._battery_level))
|
|
|
|
|
|
|
|
@property
|
|
|
|
def battery_icon(self):
|
2020-04-03 16:05:58 +00:00
|
|
|
"""Return the battery icon for the vacuum cleaner.
|
|
|
|
|
|
|
|
No need to check SUPPORT_BATTERY, this won't be called if battery_level is None.
|
|
|
|
"""
|
2017-09-15 13:39:20 +00:00
|
|
|
return icon_for_battery_level(
|
2019-07-31 19:25:30 +00:00
|
|
|
battery_level=self.battery_level, charging=self._charging
|
|
|
|
)
|
2017-09-15 13:39:20 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Flag supported features."""
|
|
|
|
return self._supported_features
|
|
|
|
|
2018-10-01 06:55:00 +00:00
|
|
|
async def async_turn_on(self, **kwargs):
|
2017-09-15 13:39:20 +00:00
|
|
|
"""Turn the vacuum on."""
|
|
|
|
if self.supported_features & SUPPORT_TURN_ON == 0:
|
|
|
|
return
|
|
|
|
|
2021-10-28 06:13:32 +00:00
|
|
|
await mqtt.async_publish(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass,
|
|
|
|
self._command_topic,
|
|
|
|
self._payloads[CONF_PAYLOAD_TURN_ON],
|
|
|
|
self._qos,
|
|
|
|
self._retain,
|
2022-01-03 08:03:47 +00:00
|
|
|
self._encoding,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
self._status = "Cleaning"
|
2019-03-12 21:46:48 +00:00
|
|
|
self.async_write_ha_state()
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2018-10-01 06:55:00 +00:00
|
|
|
async def async_turn_off(self, **kwargs):
|
2017-09-15 13:39:20 +00:00
|
|
|
"""Turn the vacuum off."""
|
|
|
|
if self.supported_features & SUPPORT_TURN_OFF == 0:
|
2019-04-22 19:49:15 +00:00
|
|
|
return None
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2021-10-28 06:13:32 +00:00
|
|
|
await mqtt.async_publish(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass,
|
|
|
|
self._command_topic,
|
|
|
|
self._payloads[CONF_PAYLOAD_TURN_OFF],
|
|
|
|
self._qos,
|
|
|
|
self._retain,
|
2022-01-03 08:03:47 +00:00
|
|
|
self._encoding,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
self._status = "Turning Off"
|
2019-03-12 21:46:48 +00:00
|
|
|
self.async_write_ha_state()
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2018-10-01 06:55:00 +00:00
|
|
|
async def async_stop(self, **kwargs):
|
2017-09-15 13:39:20 +00:00
|
|
|
"""Stop the vacuum."""
|
|
|
|
if self.supported_features & SUPPORT_STOP == 0:
|
2019-04-22 19:49:15 +00:00
|
|
|
return None
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2021-10-28 06:13:32 +00:00
|
|
|
await mqtt.async_publish(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass,
|
|
|
|
self._command_topic,
|
|
|
|
self._payloads[CONF_PAYLOAD_STOP],
|
|
|
|
self._qos,
|
|
|
|
self._retain,
|
2022-01-03 08:03:47 +00:00
|
|
|
self._encoding,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
self._status = "Stopping the current task"
|
2019-03-12 21:46:48 +00:00
|
|
|
self.async_write_ha_state()
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2018-10-01 06:55:00 +00:00
|
|
|
async def async_clean_spot(self, **kwargs):
|
2017-09-15 13:39:20 +00:00
|
|
|
"""Perform a spot clean-up."""
|
|
|
|
if self.supported_features & SUPPORT_CLEAN_SPOT == 0:
|
2019-04-22 19:49:15 +00:00
|
|
|
return None
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2021-10-28 06:13:32 +00:00
|
|
|
await mqtt.async_publish(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass,
|
|
|
|
self._command_topic,
|
|
|
|
self._payloads[CONF_PAYLOAD_CLEAN_SPOT],
|
|
|
|
self._qos,
|
|
|
|
self._retain,
|
2022-01-03 08:03:47 +00:00
|
|
|
self._encoding,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2017-09-15 13:39:20 +00:00
|
|
|
self._status = "Cleaning spot"
|
2019-03-12 21:46:48 +00:00
|
|
|
self.async_write_ha_state()
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2018-10-01 06:55:00 +00:00
|
|
|
async def async_locate(self, **kwargs):
|
2017-09-15 13:39:20 +00:00
|
|
|
"""Locate the vacuum (usually by playing a song)."""
|
|
|
|
if self.supported_features & SUPPORT_LOCATE == 0:
|
2019-04-22 19:49:15 +00:00
|
|
|
return None
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2021-10-28 06:13:32 +00:00
|
|
|
await mqtt.async_publish(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass,
|
|
|
|
self._command_topic,
|
|
|
|
self._payloads[CONF_PAYLOAD_LOCATE],
|
|
|
|
self._qos,
|
|
|
|
self._retain,
|
2022-01-03 08:03:47 +00:00
|
|
|
self._encoding,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2017-09-15 13:39:20 +00:00
|
|
|
self._status = "Hi, I'm over here!"
|
2019-03-12 21:46:48 +00:00
|
|
|
self.async_write_ha_state()
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2018-10-01 06:55:00 +00:00
|
|
|
async def async_start_pause(self, **kwargs):
|
2017-09-15 13:39:20 +00:00
|
|
|
"""Start, pause or resume the cleaning task."""
|
|
|
|
if self.supported_features & SUPPORT_PAUSE == 0:
|
2019-04-22 19:49:15 +00:00
|
|
|
return None
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2021-10-28 06:13:32 +00:00
|
|
|
await mqtt.async_publish(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass,
|
|
|
|
self._command_topic,
|
|
|
|
self._payloads[CONF_PAYLOAD_START_PAUSE],
|
|
|
|
self._qos,
|
|
|
|
self._retain,
|
2022-01-03 08:03:47 +00:00
|
|
|
self._encoding,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
self._status = "Pausing/Resuming cleaning..."
|
2019-03-12 21:46:48 +00:00
|
|
|
self.async_write_ha_state()
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2018-10-01 06:55:00 +00:00
|
|
|
async def async_return_to_base(self, **kwargs):
|
2017-09-15 13:39:20 +00:00
|
|
|
"""Tell the vacuum to return to its dock."""
|
|
|
|
if self.supported_features & SUPPORT_RETURN_HOME == 0:
|
2019-04-22 19:49:15 +00:00
|
|
|
return None
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2021-10-28 06:13:32 +00:00
|
|
|
await mqtt.async_publish(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass,
|
|
|
|
self._command_topic,
|
|
|
|
self._payloads[CONF_PAYLOAD_RETURN_TO_BASE],
|
|
|
|
self._qos,
|
|
|
|
self._retain,
|
2022-01-03 08:03:47 +00:00
|
|
|
self._encoding,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
self._status = "Returning home..."
|
2019-03-12 21:46:48 +00:00
|
|
|
self.async_write_ha_state()
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2018-10-01 06:55:00 +00:00
|
|
|
async def async_set_fan_speed(self, fan_speed, **kwargs):
|
2017-09-15 13:39:20 +00:00
|
|
|
"""Set fan speed."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if (
|
|
|
|
self.supported_features & SUPPORT_FAN_SPEED == 0
|
|
|
|
) or fan_speed not in self._fan_speed_list:
|
2019-04-22 19:49:15 +00:00
|
|
|
return None
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2021-10-28 06:13:32 +00:00
|
|
|
await mqtt.async_publish(
|
2022-01-03 08:03:47 +00:00
|
|
|
self.hass,
|
|
|
|
self._set_fan_speed_topic,
|
|
|
|
fan_speed,
|
|
|
|
self._qos,
|
|
|
|
self._retain,
|
|
|
|
self._encoding,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-09-03 19:14:00 +00:00
|
|
|
self._status = f"Setting fan to {fan_speed}..."
|
2019-03-12 21:46:48 +00:00
|
|
|
self.async_write_ha_state()
|
2017-09-15 13:39:20 +00:00
|
|
|
|
2018-10-01 06:55:00 +00:00
|
|
|
async def async_send_command(self, command, params=None, **kwargs):
|
2017-09-15 13:39:20 +00:00
|
|
|
"""Send a command to a vacuum cleaner."""
|
|
|
|
if self.supported_features & SUPPORT_SEND_COMMAND == 0:
|
|
|
|
return
|
2019-04-14 18:09:46 +00:00
|
|
|
if params:
|
|
|
|
message = {"command": command}
|
|
|
|
message.update(params)
|
|
|
|
message = json.dumps(message)
|
|
|
|
else:
|
|
|
|
message = command
|
2021-10-28 06:13:32 +00:00
|
|
|
await mqtt.async_publish(
|
2022-01-03 08:03:47 +00:00
|
|
|
self.hass,
|
|
|
|
self._send_command_topic,
|
|
|
|
message,
|
|
|
|
self._qos,
|
|
|
|
self._retain,
|
|
|
|
self._encoding,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-09-03 19:14:00 +00:00
|
|
|
self._status = f"Sending command {message}..."
|
2019-03-12 21:46:48 +00:00
|
|
|
self.async_write_ha_state()
|