From 76675ea82f8cff679fa6284d13f375d094a4f80c Mon Sep 17 00:00:00 2001 From: Ben Vezzani Date: Sun, 31 Mar 2024 22:26:04 -0400 Subject: [PATCH] Initial support for 360 Vis Nav devices --- custom_components/dyson_local/__init__.py | 9 ++++- .../dyson_local/binary_sensor.py | 40 ++++++++++++++++++- custom_components/dyson_local/sensor.py | 3 +- .../dyson_local/vendor/libdyson/__init__.py | 4 ++ .../vendor/libdyson/cloud/account.py | 1 + .../dyson_local/vendor/libdyson/const.py | 1 + .../vendor/libdyson/dyson_360_vis_nav.py | 13 ++++++ 7 files changed, 67 insertions(+), 4 deletions(-) create mode 100644 custom_components/dyson_local/vendor/libdyson/dyson_360_vis_nav.py diff --git a/custom_components/dyson_local/__init__.py b/custom_components/dyson_local/__init__.py index 83235c1..1ec086d 100644 --- a/custom_components/dyson_local/__init__.py +++ b/custom_components/dyson_local/__init__.py @@ -9,6 +9,7 @@ from .vendor.libdyson import ( Dyson360Eye, Dyson360Heurist, + Dyson360VisNav, DysonPureHotCool, DysonPureHotCoolLink, DysonPurifierHumidifyCool, @@ -113,7 +114,9 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: entry.data[CONF_DEVICE_TYPE], ) - if not isinstance(device, Dyson360Eye) and not isinstance(device, Dyson360Heurist): + if (not isinstance(device, Dyson360Eye) + and not isinstance(device, Dyson360Heurist) + and not isinstance(device, Dyson360VisNav)): # Set up coordinator async def async_update_data(): """Poll environmental data from the device.""" @@ -203,7 +206,9 @@ async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: @callback def _async_get_platforms(device: DysonDevice) -> List[str]: - if isinstance(device, Dyson360Eye) or isinstance(device, Dyson360Heurist): + if (isinstance(device, Dyson360Eye) + or isinstance(device, Dyson360Heurist) + or isinstance(device, Dyson360VisNav)): return ["binary_sensor", "sensor", "vacuum"] platforms = ["fan", "select", "sensor", "switch"] if isinstance(device, DysonPureHotCool): diff --git a/custom_components/dyson_local/binary_sensor.py b/custom_components/dyson_local/binary_sensor.py index 72c0073..1c6d826 100644 --- a/custom_components/dyson_local/binary_sensor.py +++ b/custom_components/dyson_local/binary_sensor.py @@ -2,7 +2,12 @@ from typing import Callable -from .vendor.libdyson import Dyson360Eye, Dyson360Heurist, DysonPureHotCoolLink +from .vendor.libdyson import ( + Dyson360Eye, + Dyson360Heurist, + Dyson360VisNav, + DysonPureHotCoolLink, +) from homeassistant.components.binary_sensor import ( BinarySensorDeviceClass, @@ -29,6 +34,13 @@ async def async_setup_entry( if isinstance(device, Dyson360Eye): entities.append(DysonVacuumBatteryChargingSensor(device, name)) if isinstance(device, Dyson360Heurist): + entities.extend( + [ + DysonVacuumBatteryChargingSensor(device, name), + g(device, name), + ] + ) + if isinstance(device, Dyson360VisNav): entities.extend( [ DysonVacuumBatteryChargingSensor(device, name), @@ -92,6 +104,32 @@ def sub_unique_id(self): return "bin_full" +class Dyson360VisNavBinFullSensor(DysonEntity, BinarySensorEntity): + """Dyson 360 VisNav bin full sensor.""" + + _attr_entity_category = EntityCategory.DIAGNOSTIC + + @property + def is_on(self) -> bool: + """Return if the sensor is on.""" + return self._device.is_bin_full + + @property + def icon(self) -> str: + """Return the sensor icon.""" + return ICON_BIN_FULL + + @property + def sub_name(self) -> str: + """Return the name of the sensor.""" + return "Bin Full" + + @property + def sub_unique_id(self): + """Return the sensor's unique id.""" + return "bin_full" + + class DysonPureHotCoolLinkTiltSensor(DysonEntity, BinarySensorEntity): """Dyson Pure Hot+Cool Link tilt sensor.""" diff --git a/custom_components/dyson_local/sensor.py b/custom_components/dyson_local/sensor.py index 8c8e8d2..06d479d 100644 --- a/custom_components/dyson_local/sensor.py +++ b/custom_components/dyson_local/sensor.py @@ -5,6 +5,7 @@ from .vendor.libdyson import ( Dyson360Eye, Dyson360Heurist, + Dyson360VisNav, DysonDevice, DysonPureCoolLink, DysonPurifierHumidifyCool, @@ -44,7 +45,7 @@ async def async_setup_entry( """Set up Dyson sensor from a config entry.""" device = hass.data[DOMAIN][DATA_DEVICES][config_entry.entry_id] name = config_entry.data[CONF_NAME] - if isinstance(device, Dyson360Eye) or isinstance(device, Dyson360Heurist): + if isinstance(device, Dyson360Eye) or isinstance(device, Dyson360Heurist) or isinstance(device, Dyson360VisNav): entities = [DysonBatterySensor(device, name)] else: coordinator = hass.data[DOMAIN][DATA_COORDINATORS][config_entry.entry_id] diff --git a/custom_components/dyson_local/vendor/libdyson/__init__.py b/custom_components/dyson_local/vendor/libdyson/__init__.py index 1618ff4..6dda875 100644 --- a/custom_components/dyson_local/vendor/libdyson/__init__.py +++ b/custom_components/dyson_local/vendor/libdyson/__init__.py @@ -4,6 +4,7 @@ from .const import ( DEVICE_TYPE_360_EYE, DEVICE_TYPE_360_HEURIST, + DEVICE_TYPE_360_VIS_NAV, DEVICE_TYPE_PURE_COOL, DEVICE_TYPE_PURIFIER_COOL_E, DEVICE_TYPE_PURIFIER_COOL_K, @@ -32,6 +33,7 @@ from .discovery import DysonDiscovery # noqa: F401 from .dyson_360_eye import Dyson360Eye from .dyson_360_heurist import Dyson360Heurist +from .dyson_360_vis_nav import Dyson360VisNav from .dyson_device import DysonDevice from .dyson_pure_cool import DysonPureCool from .dyson_pure_cool_link import DysonPureCoolLink @@ -48,6 +50,8 @@ def get_device(serial: str, credential: str, device_type: str) -> Optional[Dyson return Dyson360Eye(serial, credential) if device_type == DEVICE_TYPE_360_HEURIST: return Dyson360Heurist(serial, credential) + if device_type == DEVICE_TYPE_360_VIS_NAV: + return Dyson360VisNav(serial, credential) if device_type in [ DEVICE_TYPE_PURE_COOL_LINK_DESK, DEVICE_TYPE_PURE_COOL_LINK, diff --git a/custom_components/dyson_local/vendor/libdyson/cloud/account.py b/custom_components/dyson_local/vendor/libdyson/cloud/account.py index c2f0d0b..4ce3378 100644 --- a/custom_components/dyson_local/vendor/libdyson/cloud/account.py +++ b/custom_components/dyson_local/vendor/libdyson/cloud/account.py @@ -35,6 +35,7 @@ FILE_PATH = pathlib.Path(__file__).parent.absolute() + class HTTPBearerAuth(AuthBase): """Attaches HTTP Bearder Authentication to the given Request object.""" diff --git a/custom_components/dyson_local/vendor/libdyson/const.py b/custom_components/dyson_local/vendor/libdyson/const.py index 2597888..534d925 100644 --- a/custom_components/dyson_local/vendor/libdyson/const.py +++ b/custom_components/dyson_local/vendor/libdyson/const.py @@ -3,6 +3,7 @@ DEVICE_TYPE_360_EYE = "N223" DEVICE_TYPE_360_HEURIST = "276" +DEVICE_TYPE_360_VIS_NAV = "277" DEVICE_TYPE_PURE_COOL_LINK_DESK = "469" # DP01? DP02? This one's a bit older, and scraping the Dyson website is unclear DEVICE_TYPE_PURE_COOL_DESK = "520" # AM06? This one's also a bit older, and also hard to scrape off the Dyson website DEVICE_TYPE_PURE_COOL_LINK = "475" # TP02 diff --git a/custom_components/dyson_local/vendor/libdyson/dyson_360_vis_nav.py b/custom_components/dyson_local/vendor/libdyson/dyson_360_vis_nav.py new file mode 100644 index 0000000..f2e3b30 --- /dev/null +++ b/custom_components/dyson_local/vendor/libdyson/dyson_360_vis_nav.py @@ -0,0 +1,13 @@ +"""Dyson 360 Vis Nav vacuum robot.""" + +from .const import DEVICE_TYPE_360_VIS_NAV +from .dyson_360_heurist import Dyson360Heurist + + +class Dyson360VisNav(Dyson360Heurist): + """Dyson 360 Vis Nav device.""" + + @property + def device_type(self) -> str: + """Return the device type.""" + return DEVICE_TYPE_360_VIS_NAV