432 lines
15 KiB
Python
432 lines
15 KiB
Python
"""Support for Neato Connected Vacuums."""
|
|
from __future__ import annotations
|
|
|
|
from datetime import timedelta
|
|
import logging
|
|
from typing import Any
|
|
|
|
from pybotvac import Robot
|
|
from pybotvac.exceptions import NeatoRobotException
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.components.vacuum import (
|
|
ATTR_STATUS,
|
|
STATE_CLEANING,
|
|
STATE_DOCKED,
|
|
STATE_ERROR,
|
|
STATE_RETURNING,
|
|
SUPPORT_BATTERY,
|
|
SUPPORT_CLEAN_SPOT,
|
|
SUPPORT_LOCATE,
|
|
SUPPORT_MAP,
|
|
SUPPORT_PAUSE,
|
|
SUPPORT_RETURN_HOME,
|
|
SUPPORT_START,
|
|
SUPPORT_STATE,
|
|
SUPPORT_STOP,
|
|
StateVacuumEntity,
|
|
)
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import ATTR_MODE, STATE_IDLE, STATE_PAUSED
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers import config_validation as cv, entity_platform
|
|
from homeassistant.helpers.entity import DeviceInfo
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from .const import (
|
|
ACTION,
|
|
ALERTS,
|
|
ERRORS,
|
|
MODE,
|
|
NEATO_DOMAIN,
|
|
NEATO_LOGIN,
|
|
NEATO_MAP_DATA,
|
|
NEATO_PERSISTENT_MAPS,
|
|
NEATO_ROBOTS,
|
|
SCAN_INTERVAL_MINUTES,
|
|
)
|
|
from .hub import NeatoHub
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
SCAN_INTERVAL = timedelta(minutes=SCAN_INTERVAL_MINUTES)
|
|
|
|
SUPPORT_NEATO = (
|
|
SUPPORT_BATTERY
|
|
| SUPPORT_PAUSE
|
|
| SUPPORT_RETURN_HOME
|
|
| SUPPORT_STOP
|
|
| SUPPORT_START
|
|
| SUPPORT_CLEAN_SPOT
|
|
| SUPPORT_STATE
|
|
| SUPPORT_MAP
|
|
| SUPPORT_LOCATE
|
|
)
|
|
|
|
ATTR_CLEAN_START = "clean_start"
|
|
ATTR_CLEAN_STOP = "clean_stop"
|
|
ATTR_CLEAN_AREA = "clean_area"
|
|
ATTR_CLEAN_BATTERY_START = "battery_level_at_clean_start"
|
|
ATTR_CLEAN_BATTERY_END = "battery_level_at_clean_end"
|
|
ATTR_CLEAN_SUSP_COUNT = "clean_suspension_count"
|
|
ATTR_CLEAN_SUSP_TIME = "clean_suspension_time"
|
|
ATTR_CLEAN_PAUSE_TIME = "clean_pause_time"
|
|
ATTR_CLEAN_ERROR_TIME = "clean_error_time"
|
|
ATTR_LAUNCHED_FROM = "launched_from"
|
|
|
|
ATTR_NAVIGATION = "navigation"
|
|
ATTR_CATEGORY = "category"
|
|
ATTR_ZONE = "zone"
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
|
) -> None:
|
|
"""Set up Neato vacuum with config entry."""
|
|
dev = []
|
|
neato: NeatoHub = hass.data[NEATO_LOGIN]
|
|
mapdata: dict[str, Any] | None = hass.data.get(NEATO_MAP_DATA)
|
|
persistent_maps: dict[str, Any] | None = hass.data.get(NEATO_PERSISTENT_MAPS)
|
|
for robot in hass.data[NEATO_ROBOTS]:
|
|
dev.append(NeatoConnectedVacuum(neato, robot, mapdata, persistent_maps))
|
|
|
|
if not dev:
|
|
return
|
|
|
|
_LOGGER.debug("Adding vacuums %s", dev)
|
|
async_add_entities(dev, True)
|
|
|
|
platform = entity_platform.async_get_current_platform()
|
|
assert platform is not None
|
|
|
|
platform.async_register_entity_service(
|
|
"custom_cleaning",
|
|
{
|
|
vol.Optional(ATTR_MODE, default=2): cv.positive_int,
|
|
vol.Optional(ATTR_NAVIGATION, default=1): cv.positive_int,
|
|
vol.Optional(ATTR_CATEGORY, default=4): cv.positive_int,
|
|
vol.Optional(ATTR_ZONE): cv.string,
|
|
},
|
|
"neato_custom_cleaning",
|
|
)
|
|
|
|
|
|
class NeatoConnectedVacuum(StateVacuumEntity):
|
|
"""Representation of a Neato Connected Vacuum."""
|
|
|
|
def __init__(
|
|
self,
|
|
neato: NeatoHub,
|
|
robot: Robot,
|
|
mapdata: dict[str, Any] | None,
|
|
persistent_maps: dict[str, Any] | None,
|
|
) -> None:
|
|
"""Initialize the Neato Connected Vacuum."""
|
|
self.robot = robot
|
|
self._available: bool = neato is not None
|
|
self._mapdata = mapdata
|
|
self._name: str = f"{self.robot.name}"
|
|
self._robot_has_map: bool = self.robot.has_persistent_maps
|
|
self._robot_maps = persistent_maps
|
|
self._robot_serial: str = self.robot.serial
|
|
self._status_state: str | None = None
|
|
self._clean_state: str | None = None
|
|
self._state: dict[str, Any] | None = None
|
|
self._clean_time_start: str | None = None
|
|
self._clean_time_stop: str | None = None
|
|
self._clean_area: float | None = None
|
|
self._clean_battery_start: int | None = None
|
|
self._clean_battery_end: int | None = None
|
|
self._clean_susp_charge_count: int | None = None
|
|
self._clean_susp_time: int | None = None
|
|
self._clean_pause_time: int | None = None
|
|
self._clean_error_time: int | None = None
|
|
self._launched_from: str | None = None
|
|
self._battery_level: int | None = None
|
|
self._robot_boundaries: list = []
|
|
self._robot_stats: dict[str, Any] | None = None
|
|
|
|
def update(self) -> None:
|
|
"""Update the states of Neato Vacuums."""
|
|
_LOGGER.debug("Running Neato Vacuums update for '%s'", self.entity_id)
|
|
try:
|
|
if self._robot_stats is None:
|
|
self._robot_stats = self.robot.get_general_info().json().get("data")
|
|
except NeatoRobotException:
|
|
_LOGGER.warning("Couldn't fetch robot information of %s", self.entity_id)
|
|
|
|
try:
|
|
self._state = self.robot.state
|
|
except NeatoRobotException as ex:
|
|
if self._available: # print only once when available
|
|
_LOGGER.error(
|
|
"Neato vacuum connection error for '%s': %s", self.entity_id, ex
|
|
)
|
|
self._state = None
|
|
self._available = False
|
|
return
|
|
|
|
if self._state is None:
|
|
return
|
|
self._available = True
|
|
_LOGGER.debug("self._state=%s", self._state)
|
|
if "alert" in self._state:
|
|
robot_alert = ALERTS.get(self._state["alert"])
|
|
else:
|
|
robot_alert = None
|
|
if self._state["state"] == 1:
|
|
if self._state["details"]["isCharging"]:
|
|
self._clean_state = STATE_DOCKED
|
|
self._status_state = "Charging"
|
|
elif (
|
|
self._state["details"]["isDocked"]
|
|
and not self._state["details"]["isCharging"]
|
|
):
|
|
self._clean_state = STATE_DOCKED
|
|
self._status_state = "Docked"
|
|
else:
|
|
self._clean_state = STATE_IDLE
|
|
self._status_state = "Stopped"
|
|
|
|
if robot_alert is not None:
|
|
self._status_state = robot_alert
|
|
elif self._state["state"] == 2:
|
|
if robot_alert is None:
|
|
self._clean_state = STATE_CLEANING
|
|
self._status_state = (
|
|
f"{MODE.get(self._state['cleaning']['mode'])} "
|
|
f"{ACTION.get(self._state['action'])}"
|
|
)
|
|
if (
|
|
"boundary" in self._state["cleaning"]
|
|
and "name" in self._state["cleaning"]["boundary"]
|
|
):
|
|
self._status_state += (
|
|
f" {self._state['cleaning']['boundary']['name']}"
|
|
)
|
|
else:
|
|
self._status_state = robot_alert
|
|
elif self._state["state"] == 3:
|
|
self._clean_state = STATE_PAUSED
|
|
self._status_state = "Paused"
|
|
elif self._state["state"] == 4:
|
|
self._clean_state = STATE_ERROR
|
|
self._status_state = ERRORS.get(self._state["error"])
|
|
|
|
self._battery_level = self._state["details"]["charge"]
|
|
|
|
if self._mapdata is None or not self._mapdata.get(self._robot_serial, {}).get(
|
|
"maps", []
|
|
):
|
|
return
|
|
|
|
mapdata: dict[str, Any] = self._mapdata[self._robot_serial]["maps"][0]
|
|
self._clean_time_start = mapdata["start_at"]
|
|
self._clean_time_stop = mapdata["end_at"]
|
|
self._clean_area = mapdata["cleaned_area"]
|
|
self._clean_susp_charge_count = mapdata["suspended_cleaning_charging_count"]
|
|
self._clean_susp_time = mapdata["time_in_suspended_cleaning"]
|
|
self._clean_pause_time = mapdata["time_in_pause"]
|
|
self._clean_error_time = mapdata["time_in_error"]
|
|
self._clean_battery_start = mapdata["run_charge_at_start"]
|
|
self._clean_battery_end = mapdata["run_charge_at_end"]
|
|
self._launched_from = mapdata["launched_from"]
|
|
|
|
if (
|
|
self._robot_has_map
|
|
and self._state
|
|
and self._state["availableServices"]["maps"] != "basic-1"
|
|
and self._robot_maps
|
|
):
|
|
allmaps: dict = self._robot_maps[self._robot_serial]
|
|
_LOGGER.debug(
|
|
"Found the following maps for '%s': %s", self.entity_id, allmaps
|
|
)
|
|
self._robot_boundaries = [] # Reset boundaries before refreshing boundaries
|
|
for maps in allmaps:
|
|
try:
|
|
robot_boundaries = self.robot.get_map_boundaries(maps["id"]).json()
|
|
except NeatoRobotException as ex:
|
|
_LOGGER.error(
|
|
"Could not fetch map boundaries for '%s': %s",
|
|
self.entity_id,
|
|
ex,
|
|
)
|
|
return
|
|
|
|
_LOGGER.debug(
|
|
"Boundaries for robot '%s' in map '%s': %s",
|
|
self.entity_id,
|
|
maps["name"],
|
|
robot_boundaries,
|
|
)
|
|
if "boundaries" in robot_boundaries["data"]:
|
|
self._robot_boundaries += robot_boundaries["data"]["boundaries"]
|
|
_LOGGER.debug(
|
|
"List of boundaries for '%s': %s",
|
|
self.entity_id,
|
|
self._robot_boundaries,
|
|
)
|
|
|
|
@property
|
|
def name(self) -> str:
|
|
"""Return the name of the device."""
|
|
return self._name
|
|
|
|
@property
|
|
def supported_features(self) -> int:
|
|
"""Flag vacuum cleaner robot features that are supported."""
|
|
return SUPPORT_NEATO
|
|
|
|
@property
|
|
def battery_level(self) -> int | None:
|
|
"""Return the battery level of the vacuum cleaner."""
|
|
return self._battery_level
|
|
|
|
@property
|
|
def available(self) -> bool:
|
|
"""Return if the robot is available."""
|
|
return self._available
|
|
|
|
@property
|
|
def icon(self) -> str:
|
|
"""Return neato specific icon."""
|
|
return "mdi:robot-vacuum-variant"
|
|
|
|
@property
|
|
def state(self) -> str | None:
|
|
"""Return the status of the vacuum cleaner."""
|
|
return self._clean_state
|
|
|
|
@property
|
|
def unique_id(self) -> str:
|
|
"""Return a unique ID."""
|
|
return self._robot_serial
|
|
|
|
@property
|
|
def extra_state_attributes(self) -> dict[str, Any]:
|
|
"""Return the state attributes of the vacuum cleaner."""
|
|
data: dict[str, Any] = {}
|
|
|
|
if self._status_state is not None:
|
|
data[ATTR_STATUS] = self._status_state
|
|
if self._clean_time_start is not None:
|
|
data[ATTR_CLEAN_START] = self._clean_time_start
|
|
if self._clean_time_stop is not None:
|
|
data[ATTR_CLEAN_STOP] = self._clean_time_stop
|
|
if self._clean_area is not None:
|
|
data[ATTR_CLEAN_AREA] = self._clean_area
|
|
if self._clean_susp_charge_count is not None:
|
|
data[ATTR_CLEAN_SUSP_COUNT] = self._clean_susp_charge_count
|
|
if self._clean_susp_time is not None:
|
|
data[ATTR_CLEAN_SUSP_TIME] = self._clean_susp_time
|
|
if self._clean_pause_time is not None:
|
|
data[ATTR_CLEAN_PAUSE_TIME] = self._clean_pause_time
|
|
if self._clean_error_time is not None:
|
|
data[ATTR_CLEAN_ERROR_TIME] = self._clean_error_time
|
|
if self._clean_battery_start is not None:
|
|
data[ATTR_CLEAN_BATTERY_START] = self._clean_battery_start
|
|
if self._clean_battery_end is not None:
|
|
data[ATTR_CLEAN_BATTERY_END] = self._clean_battery_end
|
|
if self._launched_from is not None:
|
|
data[ATTR_LAUNCHED_FROM] = self._launched_from
|
|
|
|
return data
|
|
|
|
@property
|
|
def device_info(self) -> DeviceInfo:
|
|
"""Device info for neato robot."""
|
|
stats = self._robot_stats
|
|
return DeviceInfo(
|
|
identifiers={(NEATO_DOMAIN, self._robot_serial)},
|
|
manufacturer=stats["battery"]["vendor"] if stats else None,
|
|
model=stats["model"] if stats else None,
|
|
name=self._name,
|
|
sw_version=stats["firmware"] if stats else None,
|
|
)
|
|
|
|
def start(self) -> None:
|
|
"""Start cleaning or resume cleaning."""
|
|
if self._state:
|
|
try:
|
|
if self._state["state"] == 1:
|
|
self.robot.start_cleaning()
|
|
elif self._state["state"] == 3:
|
|
self.robot.resume_cleaning()
|
|
except NeatoRobotException as ex:
|
|
_LOGGER.error(
|
|
"Neato vacuum connection error for '%s': %s", self.entity_id, ex
|
|
)
|
|
|
|
def pause(self) -> None:
|
|
"""Pause the vacuum."""
|
|
try:
|
|
self.robot.pause_cleaning()
|
|
except NeatoRobotException as ex:
|
|
_LOGGER.error(
|
|
"Neato vacuum connection error for '%s': %s", self.entity_id, ex
|
|
)
|
|
|
|
def return_to_base(self, **kwargs: Any) -> None:
|
|
"""Set the vacuum cleaner to return to the dock."""
|
|
try:
|
|
if self._clean_state == STATE_CLEANING:
|
|
self.robot.pause_cleaning()
|
|
self._clean_state = STATE_RETURNING
|
|
self.robot.send_to_base()
|
|
except NeatoRobotException as ex:
|
|
_LOGGER.error(
|
|
"Neato vacuum connection error for '%s': %s", self.entity_id, ex
|
|
)
|
|
|
|
def stop(self, **kwargs: Any) -> None:
|
|
"""Stop the vacuum cleaner."""
|
|
try:
|
|
self.robot.stop_cleaning()
|
|
except NeatoRobotException as ex:
|
|
_LOGGER.error(
|
|
"Neato vacuum connection error for '%s': %s", self.entity_id, ex
|
|
)
|
|
|
|
def locate(self, **kwargs: Any) -> None:
|
|
"""Locate the robot by making it emit a sound."""
|
|
try:
|
|
self.robot.locate()
|
|
except NeatoRobotException as ex:
|
|
_LOGGER.error(
|
|
"Neato vacuum connection error for '%s': %s", self.entity_id, ex
|
|
)
|
|
|
|
def clean_spot(self, **kwargs: Any) -> None:
|
|
"""Run a spot cleaning starting from the base."""
|
|
try:
|
|
self.robot.start_spot_cleaning()
|
|
except NeatoRobotException as ex:
|
|
_LOGGER.error(
|
|
"Neato vacuum connection error for '%s': %s", self.entity_id, ex
|
|
)
|
|
|
|
def neato_custom_cleaning(
|
|
self, mode: str, navigation: str, category: str, zone: str | None = None
|
|
) -> None:
|
|
"""Zone cleaning service call."""
|
|
boundary_id = None
|
|
if zone is not None:
|
|
for boundary in self._robot_boundaries:
|
|
if zone in boundary["name"]:
|
|
boundary_id = boundary["id"]
|
|
if boundary_id is None:
|
|
_LOGGER.error(
|
|
"Zone '%s' was not found for the robot '%s'", zone, self.entity_id
|
|
)
|
|
return
|
|
_LOGGER.info("Start cleaning zone '%s' with robot %s", zone, self.entity_id)
|
|
|
|
self._clean_state = STATE_CLEANING
|
|
try:
|
|
self.robot.start_cleaning(mode, navigation, category, boundary_id)
|
|
except NeatoRobotException as ex:
|
|
_LOGGER.error(
|
|
"Neato vacuum connection error for '%s': %s", self.entity_id, ex
|
|
)
|