| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123 | """Library to handle connection with Switchbot."""from __future__ import annotationsimport loggingfrom typing import Anyfrom .device import (    DEVICE_SET_EXTENDED_KEY,    DEVICE_SET_MODE_KEY,    SwitchbotDeviceOverrideStateDuringConnection,    update_after_operation,)_LOGGER = logging.getLogger(__name__)BOT_COMMAND_HEADER = "5701"# Bot keysPRESS_KEY = f"{BOT_COMMAND_HEADER}00"ON_KEY = f"{BOT_COMMAND_HEADER}01"OFF_KEY = f"{BOT_COMMAND_HEADER}02"DOWN_KEY = f"{BOT_COMMAND_HEADER}03"UP_KEY = f"{BOT_COMMAND_HEADER}04"class Switchbot(SwitchbotDeviceOverrideStateDuringConnection):    """Representation of a Switchbot."""    def __init__(self, *args: Any, **kwargs: Any) -> None:        """Switchbot Bot/WoHand constructor."""        super().__init__(*args, **kwargs)        self._inverse: bool = kwargs.pop("inverse_mode", False)    @update_after_operation    async def turn_on(self) -> bool:        """Turn device on."""        result = await self._send_command(ON_KEY)        ret = self._check_command_result(result, 0, {1, 5})        self._override_state({"isOn": True})        _LOGGER.debug(            "%s: Turn on result: %s -> %s",            self.name,            result.hex() if result else None,            self._override_adv_data,        )        self._fire_callbacks()        return ret    @update_after_operation    async def turn_off(self) -> bool:        """Turn device off."""        result = await self._send_command(OFF_KEY)        ret = self._check_command_result(result, 0, {1, 5})        self._override_state({"isOn": False})        _LOGGER.debug(            "%s: Turn off result: %s -> %s",            self.name,            result.hex() if result else None,            self._override_adv_data,        )        self._fire_callbacks()        return ret    @update_after_operation    async def hand_up(self) -> bool:        """Raise device arm."""        result = await self._send_command(UP_KEY)        return self._check_command_result(result, 0, {1, 5})    @update_after_operation    async def hand_down(self) -> bool:        """Lower device arm."""        result = await self._send_command(DOWN_KEY)        return self._check_command_result(result, 0, {1, 5})    @update_after_operation    async def press(self) -> bool:        """Press command to device."""        result = await self._send_command(PRESS_KEY)        return self._check_command_result(result, 0, {1, 5})    @update_after_operation    async def set_switch_mode(        self, switch_mode: bool = False, strength: int = 100, inverse: bool = False    ) -> bool:        """Change bot mode."""        mode_key = format(switch_mode, "b") + format(inverse, "b")        strength_key = f"{strength:0{2}x}"  # to hex with padding to double digit        result = await self._send_command(DEVICE_SET_MODE_KEY + strength_key + mode_key)        return self._check_command_result(result, 0, {1})    @update_after_operation    async def set_long_press(self, duration: int = 0) -> bool:        """Set bot long press duration."""        duration_key = f"{duration:0{2}x}"  # to hex with padding to double digit        result = await self._send_command(DEVICE_SET_EXTENDED_KEY + "08" + duration_key)        return self._check_command_result(result, 0, {1})    async def get_basic_info(self) -> dict[str, Any] | None:        """Get device basic settings."""        if not (_data := await self._get_basic_info()):            return None        return {            "battery": _data[1],            "firmware": _data[2] / 10.0,            "strength": _data[3],            "timers": _data[8],            "switchMode": bool(_data[9] & 16),            "inverseDirection": bool(_data[9] & 1),            "holdSeconds": _data[10],        }    def is_on(self) -> bool | None:        """Return switch state from cache."""        # To get actual position call update() first.        value = self._get_adv_value("isOn")        if value is None:            return None        if self._inverse:            return not value        return value
 |