* init roborock commit * init commit of roborock * removed some non-vacuum related code * removed some non-needed constants * removed translations * removed options flow * removed manual control * remove password login * removed go-to * removed unneeded function and improved device_stat * removed utils as it is unused * typing changes in vacuum.py * fixed test patch paths * removed unneeded records * removing unneeded code in tests * remove password from strings * removed maps in code * changed const, reworked functions * remove menu * fixed tests * 100% code coverage config_flow * small changes * removed unneeded patch * bump to 0.1.7 * removed services * removed extra functions and mop * add () to configEntryNotReady * moved coordinator into seperate file * update roborock testing * removed stale options code * normalize username for unique id * removed unneeded variables * fixed linter problems * removed stale comment * additional pr changes * simplify config_flow * fix config flow test * Apply suggestions from code review Co-authored-by: Allen Porter <allen.porter@gmail.com> * First pass at resolving PR comments * reworked config flow * moving vacuum attr * attempt to clean up conflig flow more * update package and use offline functionality * Fixed errors and fan bug * rework model and some other small changes * bump version * used default factory * moved some client creation into coord * fixed patch * Update homeassistant/components/roborock/coordinator.py Co-authored-by: Allen Porter <allen.porter@gmail.com> * moved async functions into gather * reworked gathers * removed random line * error catch if networking doesn't exist or timeout * bump to 0.6.5 * fixed mocked data reference url * change checking if we have no network information Co-authored-by: Allen Porter <allen.porter@gmail.com> --------- Co-authored-by: Allen Porter <allen.porter@gmail.com> Co-authored-by: Allen Porter <allen@thebends.org>
173 lines
6 KiB
Python
173 lines
6 KiB
Python
"""Support for Roborock vacuum class."""
|
|
from typing import Any
|
|
|
|
from roborock.code_mappings import RoborockFanPowerCode, RoborockStateCode
|
|
from roborock.typing import RoborockCommand
|
|
|
|
from homeassistant.components.vacuum import (
|
|
STATE_CLEANING,
|
|
STATE_DOCKED,
|
|
STATE_ERROR,
|
|
STATE_IDLE,
|
|
STATE_PAUSED,
|
|
STATE_RETURNING,
|
|
StateVacuumEntity,
|
|
VacuumEntityFeature,
|
|
)
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
from homeassistant.util import slugify
|
|
|
|
from .const import DOMAIN
|
|
from .coordinator import RoborockDataUpdateCoordinator
|
|
from .device import RoborockCoordinatedEntity
|
|
from .models import RoborockHassDeviceInfo
|
|
|
|
STATE_CODE_TO_STATE = {
|
|
RoborockStateCode["1"]: STATE_IDLE, # "Starting"
|
|
RoborockStateCode["2"]: STATE_IDLE, # "Charger disconnected"
|
|
RoborockStateCode["3"]: STATE_IDLE, # "Idle"
|
|
RoborockStateCode["4"]: STATE_CLEANING, # "Remote control active"
|
|
RoborockStateCode["5"]: STATE_CLEANING, # "Cleaning"
|
|
RoborockStateCode["6"]: STATE_RETURNING, # "Returning home"
|
|
RoborockStateCode["7"]: STATE_CLEANING, # "Manual mode"
|
|
RoborockStateCode["8"]: STATE_DOCKED, # "Charging"
|
|
RoborockStateCode["9"]: STATE_ERROR, # "Charging problem"
|
|
RoborockStateCode["10"]: STATE_PAUSED, # "Paused"
|
|
RoborockStateCode["11"]: STATE_CLEANING, # "Spot cleaning"
|
|
RoborockStateCode["12"]: STATE_ERROR, # "Error"
|
|
RoborockStateCode["13"]: STATE_IDLE, # "Shutting down"
|
|
RoborockStateCode["14"]: STATE_DOCKED, # "Updating"
|
|
RoborockStateCode["15"]: STATE_RETURNING, # "Docking"
|
|
RoborockStateCode["16"]: STATE_CLEANING, # "Going to target"
|
|
RoborockStateCode["17"]: STATE_CLEANING, # "Zoned cleaning"
|
|
RoborockStateCode["18"]: STATE_CLEANING, # "Segment cleaning"
|
|
RoborockStateCode["22"]: STATE_DOCKED, # "Emptying the bin" on s7+
|
|
RoborockStateCode["23"]: STATE_DOCKED, # "Washing the mop" on s7maxV
|
|
RoborockStateCode["26"]: STATE_RETURNING, # "Going to wash the mop" on s7maxV
|
|
RoborockStateCode["100"]: STATE_DOCKED, # "Charging complete"
|
|
RoborockStateCode["101"]: STATE_ERROR, # "Device offline"
|
|
}
|
|
|
|
|
|
ATTR_STATUS = "status"
|
|
ATTR_ERROR = "error"
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant,
|
|
config_entry: ConfigEntry,
|
|
async_add_entities: AddEntitiesCallback,
|
|
) -> None:
|
|
"""Set up the Roborock sensor."""
|
|
coordinator: RoborockDataUpdateCoordinator = hass.data[DOMAIN][
|
|
config_entry.entry_id
|
|
]
|
|
async_add_entities(
|
|
RoborockVacuum(slugify(device_id), device_info, coordinator)
|
|
for device_id, device_info in coordinator.devices_info.items()
|
|
)
|
|
|
|
|
|
class RoborockVacuum(RoborockCoordinatedEntity, StateVacuumEntity):
|
|
"""General Representation of a Roborock vacuum."""
|
|
|
|
_attr_icon = "mdi:robot-vacuum"
|
|
_attr_supported_features = (
|
|
VacuumEntityFeature.PAUSE
|
|
| VacuumEntityFeature.STOP
|
|
| VacuumEntityFeature.RETURN_HOME
|
|
| VacuumEntityFeature.FAN_SPEED
|
|
| VacuumEntityFeature.BATTERY
|
|
| VacuumEntityFeature.STATUS
|
|
| VacuumEntityFeature.SEND_COMMAND
|
|
| VacuumEntityFeature.LOCATE
|
|
| VacuumEntityFeature.CLEAN_SPOT
|
|
| VacuumEntityFeature.STATE
|
|
| VacuumEntityFeature.START
|
|
)
|
|
_attr_fan_speed_list = RoborockFanPowerCode.values()
|
|
|
|
def __init__(
|
|
self,
|
|
unique_id: str,
|
|
device: RoborockHassDeviceInfo,
|
|
coordinator: RoborockDataUpdateCoordinator,
|
|
) -> None:
|
|
"""Initialize a vacuum."""
|
|
StateVacuumEntity.__init__(self)
|
|
RoborockCoordinatedEntity.__init__(self, unique_id, device, coordinator)
|
|
|
|
@property
|
|
def state(self) -> str | None:
|
|
"""Return the status of the vacuum cleaner."""
|
|
return STATE_CODE_TO_STATE.get(self._device_status.state)
|
|
|
|
@property
|
|
def status(self) -> str | None:
|
|
"""Return the status of the vacuum cleaner."""
|
|
return self._device_status.status
|
|
|
|
@property
|
|
def battery_level(self) -> int | None:
|
|
"""Return the battery level of the vacuum cleaner."""
|
|
return self._device_status.battery
|
|
|
|
@property
|
|
def fan_speed(self) -> str | None:
|
|
"""Return the fan speed of the vacuum cleaner."""
|
|
return self._device_status.fan_power
|
|
|
|
@property
|
|
def error(self) -> str | None:
|
|
"""Get the error str if an error code exists."""
|
|
return self._device_status.error
|
|
|
|
async def async_start(self) -> None:
|
|
"""Start the vacuum."""
|
|
await self.send(RoborockCommand.APP_START)
|
|
|
|
async def async_pause(self) -> None:
|
|
"""Pause the vacuum."""
|
|
await self.send(RoborockCommand.APP_PAUSE)
|
|
|
|
async def async_stop(self, **kwargs: Any) -> None:
|
|
"""Stop the vacuum."""
|
|
await self.send(RoborockCommand.APP_STOP)
|
|
|
|
async def async_return_to_base(self, **kwargs: Any) -> None:
|
|
"""Send vacuum back to base."""
|
|
await self.send(RoborockCommand.APP_CHARGE)
|
|
|
|
async def async_clean_spot(self, **kwargs: Any) -> None:
|
|
"""Spot clean."""
|
|
await self.send(RoborockCommand.APP_SPOT)
|
|
|
|
async def async_locate(self, **kwargs: Any) -> None:
|
|
"""Locate vacuum."""
|
|
await self.send(RoborockCommand.FIND_ME)
|
|
|
|
async def async_set_fan_speed(self, fan_speed: str, **kwargs: Any) -> None:
|
|
"""Set vacuum fan speed."""
|
|
await self.send(
|
|
RoborockCommand.SET_CUSTOM_MODE,
|
|
[k for k, v in RoborockFanPowerCode.items() if v == fan_speed],
|
|
)
|
|
await self.coordinator.async_request_refresh()
|
|
|
|
async def async_start_pause(self):
|
|
"""Start, pause or resume the cleaning task."""
|
|
if self.state == STATE_CLEANING:
|
|
await self.async_pause()
|
|
else:
|
|
await self.async_start()
|
|
|
|
async def async_send_command(
|
|
self,
|
|
command: str,
|
|
params: dict[str, Any] | list[Any] | None = None,
|
|
**kwargs: Any,
|
|
) -> None:
|
|
"""Send a command to a vacuum cleaner."""
|
|
await self.send(command, params)
|