From 0f9befc649a64c09b7c4508ee007a15cff3274c6 Mon Sep 17 00:00:00 2001 From: Teemu Rytilahti Date: Sun, 2 Jun 2024 17:30:02 +0200 Subject: [PATCH] Initial support for vacuums (clean module) --- kasa/smart/modules/__init__.py | 2 + kasa/smart/modules/vacuum.py | 136 +++++++++++++++++++++++++++++++++ 2 files changed, 138 insertions(+) create mode 100644 kasa/smart/modules/vacuum.py diff --git a/kasa/smart/modules/__init__.py b/kasa/smart/modules/__init__.py index ae9fb68f..7e193f5f 100644 --- a/kasa/smart/modules/__init__.py +++ b/kasa/smart/modules/__init__.py @@ -33,6 +33,7 @@ from .temperaturesensor import TemperatureSensor from .thermostat import Thermostat from .time import Time from .triggerlogs import TriggerLogs +from .vacuum import Vacuum from .waterleaksensor import WaterleakSensor __all__ = [ @@ -66,6 +67,7 @@ __all__ = [ "TriggerLogs", "FrostProtection", "Thermostat", + "Vacuum", "SmartLightEffect", "OverheatProtection", "HomeKit", diff --git a/kasa/smart/modules/vacuum.py b/kasa/smart/modules/vacuum.py new file mode 100644 index 00000000..020d6864 --- /dev/null +++ b/kasa/smart/modules/vacuum.py @@ -0,0 +1,136 @@ +"""Implementation of vacuum (experimental).""" + +from __future__ import annotations + +import logging +from enum import IntEnum +from typing import TYPE_CHECKING + +from ...feature import Feature +from ..smartmodule import SmartModule + +if TYPE_CHECKING: + from ..smartdevice import SmartDevice + + +_LOGGER = logging.getLogger(__name__) + + +class Status(IntEnum): + """Status of vacuum.""" + + Idle = 0 + Cleaning = 1 + GoingHome = 4 + Charging = 5 + Charged = 6 + Paused = 7 + Error = 100 + Unknown = 101 + + +class Vacuum(SmartModule): + """Implementation of experimental vacuum support.""" + + REQUIRED_COMPONENT = "clean" + + def __init__(self, device: SmartDevice, module: str): + super().__init__(device, module) + self._add_feature( + Feature( + device, + "return_home", + "Return home", + container=self, + attribute_setter="return_home", + category=Feature.Category.Primary, + type=Feature.Action, + ) + ) + self._add_feature( + Feature( + device, + "start_cleaning", + "Start cleaning", + container=self, + attribute_setter="start", + category=Feature.Category.Primary, + type=Feature.Action, + ) + ) + self._add_feature( + Feature( + device, + "pause", + "Pause", + container=self, + attribute_setter="pause", + category=Feature.Category.Primary, + type=Feature.Action, + ) + ) + self._add_feature( + Feature( + device, + "status", + "Vacuum state", + container=self, + attribute_getter="status", + category=Feature.Category.Primary, + type=Feature.Type.Sensor, + ) + ) + + def query(self) -> dict: + """Query to execute during the update cycle.""" + return {"getVacStatus": None} + + async def start(self) -> None: + """Start cleaning.""" + # If we are paused, do not restart cleaning + + if self.status == Status.Paused: + return await self.resume() + + # TODO: we need to create settings for clean_modes + return self.call( + "setSwitchClean", + { + "clean_mode": 0, + "clean_on": True, + "clean_order": True, + "force_clean": False, + }, + ) + + async def pause(self): + """Pause cleaning.""" + return await self.set_pause(True) + + async def resume(self): + """Resume cleaning.""" + return await self.set_pause(False) + + async def set_pause(self, enabled: bool) -> None: + """Pause or resume cleaning.""" + return self.call("setRobotPause", {"pause": enabled}) + + async def return_home(self): + """Return home.""" + return await self.set_return_home(True) + + async def set_return_home(self, enabled: bool) -> None: + """Return home / pause returning.""" + return self.call("setSwitchCharge", {"switch_charge": enabled}) + + @property + def status(self) -> Status: + """Return current status.""" + if self.data.get("err_status"): + return Status.Error + status_code = self.data["status"] + try: + return Status(status_code) + except Exception: + _LOGGER.warning("Got unknown status code: %s (%s)", status_code, self.data) + return Status.Unknown