From 065b0f238bc5bec972467abe1cd9cf73afd96aad Mon Sep 17 00:00:00 2001 From: Ben Vezzani Date: Thu, 8 Jun 2023 10:29:17 -0400 Subject: [PATCH] Vendoring libdyson to avoid conflicts and other dependency install errors (#9) --- custom_components/dyson_cloud/__init__.py | 14 +- custom_components/dyson_cloud/camera.py | 6 +- custom_components/dyson_cloud/config_flow.py | 12 +- custom_components/dyson_cloud/manifest.json | 8 +- .../dyson_cloud/vendor/libdyson/__init__.py | 75 +++ .../vendor/libdyson/cloud/__init__.py | 7 + .../vendor/libdyson/cloud/account.py | 251 ++++++++++ .../vendor/libdyson/cloud/cloud_360_eye.py | 70 +++ .../vendor/libdyson/cloud/cloud_device.py | 13 + .../vendor/libdyson/cloud/device_info.py | 35 ++ .../vendor/libdyson/cloud/regions.py | 46 ++ .../vendor/libdyson/cloud/utils.py | 32 ++ .../dyson_cloud/vendor/libdyson/const.py | 141 ++++++ .../dyson_cloud/vendor/libdyson/discovery.py | 89 ++++ .../vendor/libdyson/dyson_360_eye.py | 29 ++ .../vendor/libdyson/dyson_360_heurist.py | 64 +++ .../vendor/libdyson/dyson_device.py | 458 ++++++++++++++++++ .../vendor/libdyson/dyson_pure_cool.py | 177 +++++++ .../vendor/libdyson/dyson_pure_cool_link.py | 93 ++++ .../vendor/libdyson/dyson_pure_hot_cool.py | 8 + .../libdyson/dyson_pure_hot_cool_link.py | 21 + .../libdyson/dyson_pure_humidify_cool.py | 102 ++++ .../vendor/libdyson/dyson_vacuum_device.py | 80 +++ .../dyson_cloud/vendor/libdyson/exceptions.py | 61 +++ .../dyson_cloud/vendor/libdyson/utils.py | 51 ++ 25 files changed, 1923 insertions(+), 20 deletions(-) create mode 100644 custom_components/dyson_cloud/vendor/libdyson/__init__.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/cloud/__init__.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/cloud/account.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/cloud/cloud_360_eye.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/cloud/cloud_device.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/cloud/device_info.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/cloud/regions.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/cloud/utils.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/const.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/discovery.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/dyson_360_eye.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/dyson_360_heurist.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/dyson_device.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/dyson_pure_cool.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/dyson_pure_cool_link.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/dyson_pure_hot_cool.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/dyson_pure_hot_cool_link.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/dyson_pure_humidify_cool.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/dyson_vacuum_device.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/exceptions.py create mode 100644 custom_components/dyson_cloud/vendor/libdyson/utils.py diff --git a/custom_components/dyson_cloud/__init__.py b/custom_components/dyson_cloud/__init__.py index 755678d..4c1eb88 100644 --- a/custom_components/dyson_cloud/__init__.py +++ b/custom_components/dyson_cloud/__init__.py @@ -5,18 +5,18 @@ from functools import partial from homeassistant.exceptions import ConfigEntryNotReady -from libdyson.cloud.account import DysonAccountCN -from libdyson.cloud.device_info import DysonDeviceInfo -from libdyson.const import DEVICE_TYPE_360_EYE -from libdyson.discovery import DysonDiscovery -from libdyson.dyson_device import DysonDevice -from libdyson.exceptions import DysonException, DysonNetworkError +from .vendor.libdyson.cloud.account import DysonAccountCN +from .vendor.libdyson.cloud.device_info import DysonDeviceInfo +from .vendor.libdyson.const import DEVICE_TYPE_360_EYE +from .vendor.libdyson.discovery import DysonDiscovery +from .vendor.libdyson.dyson_device import DysonDevice +from .vendor.libdyson.exceptions import DysonException, DysonNetworkError from homeassistant.config_entries import ConfigEntry, SOURCE_DISCOVERY from homeassistant.const import CONF_HOST, EVENT_HOMEASSISTANT_STOP from homeassistant.core import HomeAssistant, callback from homeassistant.helpers.entity import Entity from homeassistant.components.zeroconf import async_get_instance -from libdyson.cloud import DysonAccount +from .vendor.libdyson.cloud import DysonAccount from custom_components.dyson_local import DOMAIN as DYSON_LOCAL_DOMAIN from .const import CONF_AUTH, CONF_REGION, DATA_ACCOUNT, DATA_DEVICES, DOMAIN diff --git a/custom_components/dyson_cloud/camera.py b/custom_components/dyson_cloud/camera.py index 344ab7d..9bc3748 100644 --- a/custom_components/dyson_cloud/camera.py +++ b/custom_components/dyson_cloud/camera.py @@ -4,9 +4,9 @@ from homeassistant.core import HomeAssistant from homeassistant.config_entries import ConfigEntry from homeassistant.components.camera import Camera -from libdyson.const import DEVICE_TYPE_360_EYE, DEVICE_TYPE_360_HEURIST -from libdyson.cloud.cloud_360_eye import DysonCloud360Eye -from libdyson.cloud import DysonDeviceInfo +from .vendor.libdyson.const import DEVICE_TYPE_360_EYE, DEVICE_TYPE_360_HEURIST +from .vendor.libdyson.cloud.cloud_360_eye import DysonCloud360Eye +from .vendor.libdyson.cloud import DysonDeviceInfo from datetime import timedelta from .const import DATA_ACCOUNT, DATA_DEVICES, DOMAIN diff --git a/custom_components/dyson_cloud/config_flow.py b/custom_components/dyson_cloud/config_flow.py index d14b452..6595330 100644 --- a/custom_components/dyson_cloud/config_flow.py +++ b/custom_components/dyson_cloud/config_flow.py @@ -4,13 +4,13 @@ from homeassistant import config_entries from homeassistant.components.zeroconf import async_get_instance from homeassistant.const import CONF_EMAIL, CONF_HOST, CONF_PASSWORD, CONF_USERNAME -from libdyson.cloud.account import DysonAccountCN +from .vendor.libdyson.cloud.account import DysonAccountCN import voluptuous as vol -from libdyson.cloud import DysonAccount, REGIONS -from libdyson.dyson_360_eye import Dyson360Eye -from libdyson.discovery import DysonDiscovery -from libdyson.const import DEVICE_TYPE_360_EYE -from libdyson.exceptions import DysonException, DysonLoginFailure, DysonInvalidAccountStatus, DysonNetworkError, DysonOTPTooFrequently +from .vendor.libdyson.cloud import DysonAccount, REGIONS +from .vendor.libdyson.dyson_360_eye import Dyson360Eye +from .vendor.libdyson.discovery import DysonDiscovery +from .vendor.libdyson.const import DEVICE_TYPE_360_EYE +from .vendor.libdyson.exceptions import DysonException, DysonLoginFailure, DysonInvalidAccountStatus, DysonNetworkError, DysonOTPTooFrequently from voluptuous.schema_builder import Required from .const import CONF_AUTH, CONF_REGION, DOMAIN diff --git a/custom_components/dyson_cloud/manifest.json b/custom_components/dyson_cloud/manifest.json index 9485bdd..80e123a 100644 --- a/custom_components/dyson_cloud/manifest.json +++ b/custom_components/dyson_cloud/manifest.json @@ -2,10 +2,10 @@ "domain": "dyson_cloud", "name": "Dyson Cloud", "config_flow": true, - "documentation": "https://github.com/libdyson-wg/ha-dyson", - "issue_tracker": "https://github.com/libdyson-wg/ha-dyson/issues", + "documentation": "https://github.com/libdyson-wg/ha-dyson-cloud", + "issue_tracker": "https://github.com/libdyson-wg/ha-dyson-cloud/issues", "dependencies": [], "codeowners": ["@libdyson-wg"], - "requirements": ["libdyson-neon==1.0.2"], - "version": "0.18.0" + "requirements": [], + "version": "0.19.0" } diff --git a/custom_components/dyson_cloud/vendor/libdyson/__init__.py b/custom_components/dyson_cloud/vendor/libdyson/__init__.py new file mode 100644 index 0000000..47f5665 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/__init__.py @@ -0,0 +1,75 @@ +"""Dyson Python library.""" + +from typing import Optional +from .const import ( + DEVICE_TYPE_360_EYE, + DEVICE_TYPE_360_HEURIST, + DEVICE_TYPE_PURE_COOL, + DEVICE_TYPE_PURIFIER_COOL_E, + DEVICE_TYPE_PURIFIER_COOL_K, + DEVICE_TYPE_PURE_COOL_DESK, + DEVICE_TYPE_PURE_COOL_LINK, + DEVICE_TYPE_PURE_COOL_LINK_DESK, + DEVICE_TYPE_PURE_HOT_COOL, + DEVICE_TYPE_PURIFIER_HOT_COOL_E, + DEVICE_TYPE_PURIFIER_HOT_COOL_K, + DEVICE_TYPE_PURE_HOT_COOL_LINK, + DEVICE_TYPE_PURE_HUMIDIFY_COOL, + DEVICE_TYPE_PURIFIER_HUMIDIFY_COOL_E, + DEVICE_TYPE_PURIFIER_HUMIDIFY_COOL_K, +) + +from .const import CleaningMode # noqa: F401 +from .const import CleaningType # noqa: F401 +from .const import DEVICE_TYPE_NAMES # noqa: F401 +from .const import HumidifyOscillationMode # noqa: F401 +from .const import MessageType # noqa: F401 +from .const import VacuumEyePowerMode # noqa: F401 +from .const import VacuumHeuristPowerMode # noqa: F401 +from .const import VacuumState # noqa: F401 +from .const import WaterHardness # noqa: F401 +from .discovery import DysonDiscovery # noqa: F401 +from .dyson_360_eye import Dyson360Eye +from .dyson_360_heurist import Dyson360Heurist +from .dyson_device import DysonDevice +from .dyson_pure_cool import DysonPureCool +from .dyson_pure_cool_link import DysonPureCoolLink +from .dyson_pure_hot_cool import DysonPureHotCool +from .dyson_pure_hot_cool_link import DysonPureHotCoolLink +from .dyson_pure_humidify_cool import DysonPurifierHumidifyCool +from .utils import get_mqtt_info_from_wifi_info # noqa: F401 + + +def get_device(serial: str, credential: str, device_type: str) -> Optional[DysonDevice]: + """Get a new DysonDevice instance.""" + if device_type == DEVICE_TYPE_360_EYE: + return Dyson360Eye(serial, credential) + if device_type == DEVICE_TYPE_360_HEURIST: + return Dyson360Heurist(serial, credential) + if device_type in [ + DEVICE_TYPE_PURE_COOL_LINK_DESK, + DEVICE_TYPE_PURE_COOL_LINK, + ]: + return DysonPureCoolLink(serial, credential, device_type) + if device_type in [ + DEVICE_TYPE_PURE_COOL, + DEVICE_TYPE_PURIFIER_COOL_K, + DEVICE_TYPE_PURIFIER_COOL_E, + DEVICE_TYPE_PURE_COOL_DESK, + ]: + return DysonPureCool(serial, credential, device_type) + if device_type == DEVICE_TYPE_PURE_HOT_COOL_LINK: + return DysonPureHotCoolLink(serial, credential, device_type) + if device_type in [ + DEVICE_TYPE_PURE_HOT_COOL, + DEVICE_TYPE_PURIFIER_HOT_COOL_E, + DEVICE_TYPE_PURIFIER_HOT_COOL_K, + ]: + return DysonPureHotCool(serial, credential, device_type) + if device_type in [ + DEVICE_TYPE_PURE_HUMIDIFY_COOL, + DEVICE_TYPE_PURIFIER_HUMIDIFY_COOL_K, + DEVICE_TYPE_PURIFIER_HUMIDIFY_COOL_E, + ]: + return DysonPurifierHumidifyCool(serial, credential, device_type) + return None diff --git a/custom_components/dyson_cloud/vendor/libdyson/cloud/__init__.py b/custom_components/dyson_cloud/vendor/libdyson/cloud/__init__.py new file mode 100644 index 0000000..2e1eefc --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/cloud/__init__.py @@ -0,0 +1,7 @@ +"""Dyson cloud client.""" + +from .account import DysonAccount, DysonAccountCN # noqa: F401 +from .cloud_360_eye import DysonCloud360Eye # noqa: F401 +from .cloud_device import DysonCloudDevice # noqa: F401 +from .device_info import DysonDeviceInfo # noqa: F401 +from .regions import REGIONS # noqa: F401 diff --git a/custom_components/dyson_cloud/vendor/libdyson/cloud/account.py b/custom_components/dyson_cloud/vendor/libdyson/cloud/account.py new file mode 100644 index 0000000..0eaa88c --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/cloud/account.py @@ -0,0 +1,251 @@ +"""Dyson cloud account client.""" + +import pathlib +from typing import Callable, List, Optional + +import requests +from requests.auth import AuthBase, HTTPBasicAuth + +from libdyson.exceptions import ( + DysonAuthRequired, + DysonInvalidAccountStatus, + DysonInvalidAuth, + DysonLoginFailure, + DysonNetworkError, + DysonOTPTooFrequently, + DysonServerError, + DysonAPIProvisionFailure, +) + +from .device_info import DysonDeviceInfo + +DYSON_API_HOST = "https://appapi.cp.dyson.com" +DYSON_API_HOST_CN = "https://appapi.cp.dyson.cn" +DYSON_API_HEADERS = { + "User-Agent": "android client" +} + +API_PATH_PROVISION_APP = "/v1/provisioningservice/application/Android/version" +API_PATH_USER_STATUS = "/v3/userregistration/email/userstatus" +API_PATH_EMAIL_REQUEST = "/v3/userregistration/email/auth" +API_PATH_EMAIL_VERIFY = "/v3/userregistration/email/verify" +API_PATH_MOBILE_REQUEST = "/v3/userregistration/mobile/auth" +API_PATH_MOBILE_VERIFY = "/v3/userregistration/mobile/verify" +API_PATH_DEVICES = "/v2/provisioningservice/manifest" + +FILE_PATH = pathlib.Path(__file__).parent.absolute() + +class HTTPBearerAuth(AuthBase): + """Attaches HTTP Bearder Authentication to the given Request object.""" + + def __init__(self, token): + """Initialize the auth.""" + self.token = token + + def __eq__(self, other): + """Return if equal.""" + return self.token == getattr(other, "token", None) + + def __ne__(self, other): + """Return if not equal.""" + return not self == other + + def __call__(self, r): + """Attach the authentication.""" + r.headers["Authorization"] = f"Bearer {self.token}" + return r + + +class DysonAccount: + """Dyson account.""" + + _HOST = DYSON_API_HOST + + def __init__( + self, + auth_info: Optional[dict] = None, + ): + """Create a new Dyson account.""" + self._auth_info = auth_info + + @property + def auth_info(self) -> Optional[dict]: + """Return the authentication info.""" + return self._auth_info + + @property + def _auth(self) -> Optional[AuthBase]: + if self.auth_info is None: + return None + # Although basic auth is no longer used by new logins, + # we still need this for backward capability to already + # stored auth info. + if "Password" in self.auth_info: + return HTTPBasicAuth( + self.auth_info["Account"], + self.auth_info["Password"], + ) + elif self.auth_info.get("tokenType") == "Bearer": + return HTTPBearerAuth(self.auth_info["token"]) + return None + + def request( + self, + method: str, + path: str, + params: Optional[dict] = None, + data: Optional[dict] = None, + auth: bool = True, + ) -> requests.Response: + """Make API request.""" + if auth and self._auth is None: + raise DysonAuthRequired + try: + response = requests.request( + method, + self._HOST + path, + params=params, + json=data, + headers=DYSON_API_HEADERS, + auth=self._auth if auth else None, + verify=True, + ) + except requests.RequestException: + raise DysonNetworkError + if response.status_code in [401, 403]: + raise DysonInvalidAuth + if 500 <= response.status_code < 600: + raise DysonServerError + return response + + def provision_api(self) -> None: + """Provision the client connection to the API + + Calls an app provisioning API. This is expected by the Dyson App API and makes the API server ready to accept + other API calls from the current IP Address. + + Basically, this unlocks the API - the return value is not needed, and we don't need to save any cookies or + session tokens. It seems like the API Server sets some internal flag allowing API Calls from a specific address + based solely on this endpoint being called. + + This isn't likely to be a security measure. It returns a version number in a json-encoded string: `"5.0.21061"` + and that is likely consumed by an app. Presumably, an official Dyson mobile app could check the version against + some internal expected value and, for example, prompt a user that it is outdated and direct them to the app + store to download a new version in order to continue working. + """ + + response = self.request( + "GET", + API_PATH_PROVISION_APP, + params=None, + data=None, + auth=False, + ) + + if response.status_code != 200: + raise DysonAPIProvisionFailure + + def login_email_otp(self, email: str, region: str) -> Callable[[str, str], dict]: + """Login using email and OTP code.""" + self.provision_api() + + # Check account status. This tells us whether an account is active or not. + response = self.request( + "POST", + API_PATH_USER_STATUS, + params={"country": region}, + data={"email": email}, + auth=False, + ) + + jsonRes = response.json() + + account_status = jsonRes["accountStatus"] + if account_status != "ACTIVE": + raise DysonInvalidAccountStatus(account_status) + + response = self.request( + "POST", + API_PATH_EMAIL_REQUEST, + params={"country": region, "culture": "en-US"}, + data={"email": email}, + auth=False, + ) + if response.status_code == 429: + raise DysonOTPTooFrequently + + challenge_id = response.json()["challengeId"] + + def _verify(otp_code: str, password: str): + response = self.request( + "POST", + API_PATH_EMAIL_VERIFY, + data={ + "email": email, + "password": password, + "challengeId": challenge_id, + "otpCode": otp_code, + }, + auth=False, + ) + if response.status_code == 400: + raise DysonLoginFailure + body = response.json() + self._auth_info = body + return self._auth_info + + return _verify + + def devices(self) -> List[DysonDeviceInfo]: + self.provision_api() + """Get device info from cloud account.""" + devices = [] + response = self.request("GET", API_PATH_DEVICES) + for raw in response.json(): + if raw.get("LocalCredentials") is None: + # Lightcycle lights don't have LocalCredentials. + # They're not supported so just skip. + # See https://github.com/shenxn/libdyson/issues/2 for more info + continue + devices.append(DysonDeviceInfo.from_raw(raw)) + return devices + + +class DysonAccountCN(DysonAccount): + """Dyson account in Mainland China.""" + + _HOST = DYSON_API_HOST_CN + + def login_mobile_otp(self, mobile: str) -> Callable[[str], dict]: + self.provision_api() + + """Login using phone number and OTP code.""" + response = self.request( + "POST", + API_PATH_MOBILE_REQUEST, + data={"mobile": mobile}, + auth=False, + ) + if response.status_code == 429: + raise DysonOTPTooFrequently + + challenge_id = response.json()["challengeId"] + + def _verify(otp_code: str): + response = self.request( + "POST", + API_PATH_MOBILE_VERIFY, + data={ + "mobile": mobile, + "challengeId": challenge_id, + "otpCode": otp_code, + }, + auth=False, + ) + if response.status_code == 400: + raise DysonLoginFailure + body = response.json() + self._auth_info = body + return self._auth_info + + return _verify diff --git a/custom_components/dyson_cloud/vendor/libdyson/cloud/cloud_360_eye.py b/custom_components/dyson_cloud/vendor/libdyson/cloud/cloud_360_eye.py new file mode 100644 index 0000000..49e9ae8 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/cloud/cloud_360_eye.py @@ -0,0 +1,70 @@ +"""Dyson 360 Eye cloud client.""" + +from datetime import datetime, timedelta +from enum import Enum +from typing import List, Optional + +import attr + +from .cloud_device import DysonCloudDevice + + +class CleaningType(Enum): + """Cleaning type of the task.""" + + Immediate = "Immediate" + Manual = "Manual" + Scheduled = "Scheduled" + + +@attr.s(auto_attribs=True, frozen=True) +class CleaningTask: + """Represent a cleaning task.""" + + cleaning_id: str + start_time: datetime # Local time without timezone info + finish_time: datetime # Local time without timezone info + area: float # In square meters + charges: int + cleaning_type: str + is_interim: bool + + @classmethod + def from_raw(cls, raw: dict): + """Parse raw data from cloud API.""" + return cls( + raw["Clean"], + datetime.fromisoformat(raw["Started"]), + datetime.fromisoformat(raw["Finished"]), + raw["Area"], + raw["Charges"], + CleaningType(raw["Type"]), + raw["IsInterim"], + ) + + @property + def cleaning_time(self) -> timedelta: + """Return the total cleaning time.""" + return self.finish_time - self.start_time + + +class DysonCloud360Eye(DysonCloudDevice): + """Dyson 360 Eye cloud client.""" + + def get_cleaning_history(self) -> List[CleaningTask]: + """Get cleaning history from the cloud.""" + response = self._account.request( + "GET", + f"/v1/assets/devices/{self._serial}/cleanhistory", + ) + return [CleaningTask.from_raw(raw) for raw in response.json()["Entries"]] + + def get_cleaning_map(self, cleaning_id: str) -> Optional[bytes]: + """Get cleaning map in PNG format.""" + response = self._account.request( + "GET", + f"/v1/mapvisualizer/devices/{self._serial}/map/{cleaning_id}", + ) + if response.status_code == 404: + return None # No map associate with the cleaning id + return response.content diff --git a/custom_components/dyson_cloud/vendor/libdyson/cloud/cloud_device.py b/custom_components/dyson_cloud/vendor/libdyson/cloud/cloud_device.py new file mode 100644 index 0000000..15ddca7 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/cloud/cloud_device.py @@ -0,0 +1,13 @@ +"""Dyson device cloud client.""" + + +from . import DysonAccount + + +class DysonCloudDevice: + """Dyson device cloud client.""" + + def __init__(self, account: DysonAccount, serial: str): + """Initialize the client.""" + self._account = account + self._serial = serial diff --git a/custom_components/dyson_cloud/vendor/libdyson/cloud/device_info.py b/custom_components/dyson_cloud/vendor/libdyson/cloud/device_info.py new file mode 100644 index 0000000..32fce61 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/cloud/device_info.py @@ -0,0 +1,35 @@ +"""Dyson device info.""" + +from typing import Optional + +import attr + +from .utils import decrypt_password + + +@attr.s(auto_attribs=True, frozen=True) +class DysonDeviceInfo: + """Dyson device info.""" + + active: Optional[bool] + serial: str + name: str + version: str + credential: str + auto_update: bool + new_version_available: bool + product_type: str + + @classmethod + def from_raw(cls, raw: dict): + """Parse raw data.""" + return cls( + raw["Active"] if "Active" in raw else None, + raw["Serial"], + raw["Name"], + raw["Version"], + decrypt_password(raw["LocalCredentials"]), + raw["AutoUpdate"], + raw["NewVersionAvailable"], + raw["ProductType"], + ) diff --git a/custom_components/dyson_cloud/vendor/libdyson/cloud/regions.py b/custom_components/dyson_cloud/vendor/libdyson/cloud/regions.py new file mode 100644 index 0000000..8919117 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/cloud/regions.py @@ -0,0 +1,46 @@ +"""Account region list.""" + +REGIONS = { + "AU": "Australia", + "AT": "Austria", + "BE": "Belgium", + "CA": "Canada", + "HR": "Croatia", + "CZ": "Czechia", + "DK": "Denmark", + "FI": "Finland", + "FR": "France", + "DE": "Germany", + "HK": "Hong Kong", + "HU": "Hungary", + "IN": "India", + "ID": "Indonesia", + "IE": "Ireland", + "IL": "Israel", + "IT": "Italy", + "JP": "Japan", + "LT": "Lithuania", + "CN": "Mainland China", + "MY": "Malaysia", + "MX": "Mexico", + "NL": "Netherlands", + "NZ": "New Zealand", + "NO": "Norway", + "PH": "Philippines", + "PL": "Poland", + "PT": "Portugal", + "RO": "Romania", + "SA": "Saudi Arabia", + "SG": "Singapore", + "SI": "Slovenia", + "KR": "South Korea", + "ES": "Spain", + "SE": "Sweden", + "CH": "Switzerland", + "TW": "Taiwan", + "TH": "Thailand", + "TR": "Turkey", + "AE": "United Arab Emirates", + "GB": "United Kingdom", + "US": "United States of America", +} diff --git a/custom_components/dyson_cloud/vendor/libdyson/cloud/utils.py b/custom_components/dyson_cloud/vendor/libdyson/cloud/utils.py new file mode 100644 index 0000000..0e8647b --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/cloud/utils.py @@ -0,0 +1,32 @@ +"""Dyson cloud client utilities.""" + +import base64 +import json + +from cryptography.hazmat.primitives.ciphers import Cipher, algorithms, modes + +DYSON_ENCRYPTION_KEY = ( + b"\x01\x02\x03\x04\x05\x06\x07\x08\t\n\x0b\x0c\r\x0e\x0f\x10" + b"\x11\x12\x13\x14\x15\x16\x17\x18\x19\x1a\x1b\x1c\x1d\x1e\x1f " +) +DYSON_ENCRYPTION_INIT_VECTOR = ( + b"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" +) + + +def _unpad(string: str) -> str: + """Un-pad string.""" + return string[: -ord(string[len(string) - 1 :])] + + +def decrypt_password(encrypted_password: str) -> str: + """Decrypt local credential into MQTT password.""" + cipher = Cipher( + algorithms.AES(DYSON_ENCRYPTION_KEY), + modes.CBC(DYSON_ENCRYPTION_INIT_VECTOR), + ) + decryptor = cipher.decryptor() + encrypted = base64.b64decode(encrypted_password) + decrypted = decryptor.update(encrypted) + decryptor.finalize() + json_password = json.loads(_unpad(decrypted)) + return json_password["apPasswordHash"] diff --git a/custom_components/dyson_cloud/vendor/libdyson/const.py b/custom_components/dyson_cloud/vendor/libdyson/const.py new file mode 100644 index 0000000..038e735 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/const.py @@ -0,0 +1,141 @@ +"""Constants for Dyson Python library.""" +from enum import Enum, auto + +DEVICE_TYPE_360_EYE = "N223" +DEVICE_TYPE_360_HEURIST = "276" +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 +DEVICE_TYPE_PURE_COOL = "438" # TP04 +DEVICE_TYPE_PURIFIER_COOL_K = "438K" # TP07 AND TP09 +DEVICE_TYPE_PURIFIER_COOL_E = "438E" # TP07 AND TP09 +DEVICE_TYPE_PURE_HUMIDIFY_COOL = "358" # PH01 probably, but maybe PH02? Not 100% certain +DEVICE_TYPE_PURIFIER_HUMIDIFY_COOL_K = "358K" # PH03 AND PH04 +DEVICE_TYPE_PURIFIER_HUMIDIFY_COOL_E = "358E" # PH03 AND PH04 +DEVICE_TYPE_PURE_HOT_COOL_LINK = "455" # HP02 +DEVICE_TYPE_PURE_HOT_COOL = "527" # HP04 +DEVICE_TYPE_PURIFIER_HOT_COOL_E = "527E" # HP07 AND HP09 +DEVICE_TYPE_PURIFIER_HOT_COOL_K = "527K" # HP07 AND HP09 + +DEVICE_TYPE_NAMES = { + DEVICE_TYPE_360_EYE: "360 Eye robot vacuum", + DEVICE_TYPE_360_HEURIST: "360 Heurist robot vacuum", + DEVICE_TYPE_PURE_COOL: "Pure Cool", + DEVICE_TYPE_PURIFIER_COOL_K: "Purifier Cool", + DEVICE_TYPE_PURIFIER_COOL_E: "Purifier Cool", + DEVICE_TYPE_PURE_COOL_DESK: "Pure Cool Link Desk", + DEVICE_TYPE_PURE_COOL_LINK: "Pure Cool Link", + DEVICE_TYPE_PURE_COOL_LINK_DESK: "Pure Cool Link Desk", + DEVICE_TYPE_PURE_HOT_COOL: "Pure Hot+Cool", + DEVICE_TYPE_PURIFIER_HOT_COOL_E: "Pure Hot+Cool (New)", + DEVICE_TYPE_PURE_HOT_COOL_LINK: "Pure Hot+Cool Link", + DEVICE_TYPE_PURE_HUMIDIFY_COOL: "Pure Humidify+Cool", + DEVICE_TYPE_PURIFIER_HUMIDIFY_COOL_K: "Purifier Humidify+Cool", + DEVICE_TYPE_PURIFIER_HUMIDIFY_COOL_E: "Purifier Humidify+Cool", + DEVICE_TYPE_PURIFIER_HOT_COOL_K: "Purifier Hot+Cool", +} + +ENVIRONMENTAL_OFF = -1 +ENVIRONMENTAL_INIT = -2 +ENVIRONMENTAL_FAIL = -3 + + +class MessageType(Enum): + """Update message type.""" + + STATE = auto() + ENVIRONMENTAL = auto() + + +class AirQualityTarget(Enum): + """Air Quality Target.""" + + OFF = "OFF" + GOOD = "0004" + SENSITIVE = "0003" + DEFAULT = "0002" + VERY_SENSITIVE = "0001" + + +class HumidifyOscillationMode(Enum): + """Pure Humidify+Cool oscillation mode.""" + + DEGREE_45 = "0045" + DEGREE_90 = "0090" + BREEZE = "BRZE" + CUST = "CUST" + + +class WaterHardness(Enum): + """Water Hardness.""" + + SOFT = "Soft" + MEDIUM = "Medium" + HARD = "Hard" + + +class VacuumState(Enum): + """Dyson vacuum state.""" + + FAULT_CALL_HELPLINE = "FAULT_CALL_HELPLINE" + FAULT_CONTACT_HELPLINE = "FAULT_CONTACT_HELPLINE" + FAULT_CRITICAL = "FAULT_CRITICAL" + FAULT_GETTING_INFO = "FAULT_GETTING_INFO" + FAULT_LOST = "FAULT_LOST" + FAULT_ON_DOCK = "FAULT_ON_DOCK" + FAULT_ON_DOCK_CHARGED = "FAULT_ON_DOCK_CHARGED" + FAULT_ON_DOCK_CHARGING = "FAULT_ON_DOCK_CHARGING" + FAULT_REPLACE_ON_DOCK = "FAULT_REPLACE_ON_DOCK" + FAULT_RETURN_TO_DOCK = "FAULT_RETURN_TO_DOCK" + FAULT_RUNNING_DIAGNOSTIC = "FAULT_RUNNING_DIAGNOSTIC" + FAULT_USER_RECOVERABLE = "FAULT_USER_RECOVERABLE" + FULL_CLEAN_ABANDONED = "FULL_CLEAN_ABANDONED" + FULL_CLEAN_ABORTED = "FULL_CLEAN_ABORTED" + FULL_CLEAN_CHARGING = "FULL_CLEAN_CHARGING" + FULL_CLEAN_DISCOVERING = "FULL_CLEAN_DISCOVERING" + FULL_CLEAN_FINISHED = "FULL_CLEAN_FINISHED" + FULL_CLEAN_INITIATED = "FULL_CLEAN_INITIATED" + FULL_CLEAN_NEEDS_CHARGE = "FULL_CLEAN_NEEDS_CHARGE" + FULL_CLEAN_PAUSED = "FULL_CLEAN_PAUSED" + FULL_CLEAN_RUNNING = "FULL_CLEAN_RUNNING" + FULL_CLEAN_TRAVERSING = "FULL_CLEAN_TRAVERSING" + INACTIVE_CHARGED = "INACTIVE_CHARGED" + INACTIVE_CHARGING = "INACTIVE_CHARGING" + INACTIVE_DISCHARGING = "INACTIVE_DISCHARGING" + MAPPING_ABORTED = "MAPPING_ABORTED" + MAPPING_CHARGING = "MAPPING_CHARGING" + MAPPING_FINISHED = "MAPPING_FINISHED" + MAPPING_INITIATED = "MAPPING_INITIATED" + MAPPING_NEEDS_CHARGE = "MAPPING_NEEDS_CHARGE" + MAPPING_PAUSED = "MAPPING_PAUSED" + MAPPING_RUNNING = "MAPPING_RUNNING" + + +class VacuumEyePowerMode(Enum): + """Dyson 360 Eye power mode.""" + + QUIET = "halfPower" + MAX = "fullPower" + + +class VacuumHeuristPowerMode(Enum): + """Dyson 360 Heurist power mode.""" + + QUIET = "1" + HIGH = "2" + MAX = "3" + + +class CleaningType(Enum): + """Vacuum cleaning type.""" + + IMMEDIATE = "immediate" + MANUAL = "manual" + SCHEDULED = "scheduled" + + +class CleaningMode(Enum): + """Vacuum cleaning mode.""" + + GLOBAL = "global" + ZONE_CONFIGURED = "zoneConfigured" diff --git a/custom_components/dyson_cloud/vendor/libdyson/discovery.py b/custom_components/dyson_cloud/vendor/libdyson/discovery.py new file mode 100644 index 0000000..86031d4 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/discovery.py @@ -0,0 +1,89 @@ +"""Dyson device discovery.""" + +import socket +import threading +from typing import Callable, Optional + +from zeroconf import ServiceBrowser, ServiceInfo, Zeroconf + +from .dyson_device import DysonDevice + +TYPE_DYSON_360_EYE = "_360eye_mqtt._tcp.local." +TYPE_DYSON_FAN = "_dyson_mqtt._tcp.local." + + +class DysonDiscovery: + """Dyson device discovery.""" + + def __init__(self): + """Initialize the instance.""" + self._registered = {} + self._discovered = {} + self._lock = threading.Lock() + self._browser = None + + def register_device( + self, device: DysonDevice, callback: Callable[[str], None] + ) -> None: + """Register a device.""" + with self._lock: + if device.serial in self._discovered: + callback(self._discovered[device.serial]) + else: + self._registered[device.serial] = callback + + def device_discovered(self, info: ServiceInfo) -> None: + """Call when a device is discovered.""" + if info.type == TYPE_DYSON_360_EYE: + serial = (info.name.split(".")[0]).split("-", 1)[1] + else: # TYPE_DYSON_FAN + serial = (info.name.split(".")[0]).split("_")[1] + address = socket.inet_ntoa(info.addresses[0]) + with self._lock: + if serial in self._registered: + callback = self._registered.pop(serial) + callback(address) + else: + self._discovered[serial] = address + + def start_discovery(self, zeroconf_instance: Optional[Zeroconf] = None) -> None: + """Start discovery.""" + listener = DysonListener(self) + zeroconf = zeroconf_instance or Zeroconf() + self._browser = ServiceBrowser( + zeroconf, + [TYPE_DYSON_360_EYE, TYPE_DYSON_FAN], + listener, + ) + + def stop_discovery(self) -> None: + """Stop discovery.""" + try: + self._browser.cancel() + except RuntimeError: + # Throws when called from callback + # cannot join current thread + pass + self._browser.zc.close() + self._browser = None + + +class DysonListener: + """Listener for zeroconf events.""" + + def __init__(self, dyson_discovery: DysonDiscovery): + """Initialize the listener.""" + self._dyson_discovery = dyson_discovery + + def add_service(self, zeroconf: Zeroconf, type: str, name: str) -> None: + """Add a new service.""" + info = zeroconf.get_service_info(type, name) + self._dyson_discovery.device_discovered(info) + + def update_service(self, zeroconf: Zeroconf, type: str, name: str) -> None: + """Update a service.""" + # Currently not doing anything + + def remove_service(self, zeroconf: Zeroconf, type: str, name: str) -> None: + """Remove a service.""" + # Currently not doing anything diff --git a/custom_components/dyson_cloud/vendor/libdyson/dyson_360_eye.py b/custom_components/dyson_cloud/vendor/libdyson/dyson_360_eye.py new file mode 100644 index 0000000..0147c92 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/dyson_360_eye.py @@ -0,0 +1,29 @@ +"""Dyson 360 Eye vacuum robot.""" + +from .const import DEVICE_TYPE_360_EYE, VacuumEyePowerMode +from .dyson_vacuum_device import DysonVacuumDevice + + +class Dyson360Eye(DysonVacuumDevice): + """Dyson 360 Eye device.""" + + @property + def device_type(self) -> str: + """Return the device type.""" + return DEVICE_TYPE_360_EYE + + @property + def power_mode(self) -> VacuumEyePowerMode: + """Power mode of the device.""" + return VacuumEyePowerMode(self._status["currentVacuumPowerMode"]) + + def start(self) -> None: + """Start cleaning.""" + self._send_command("START", {"fullCleanType": "immediate"}) + + def set_power_mode(self, power_mode: VacuumEyePowerMode) -> None: + """Set power mode.""" + self._send_command( + "STATE-SET", + {"data": {"defaultVacuumPowerMode": power_mode.value}}, + ) diff --git a/custom_components/dyson_cloud/vendor/libdyson/dyson_360_heurist.py b/custom_components/dyson_cloud/vendor/libdyson/dyson_360_heurist.py new file mode 100644 index 0000000..70c8911 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/dyson_360_heurist.py @@ -0,0 +1,64 @@ +"""Dyson 360 Heurist vacuum robot.""" + +from typing import Optional + +from .const import DEVICE_TYPE_360_HEURIST, CleaningMode, VacuumHeuristPowerMode +from .dyson_vacuum_device import DysonVacuumDevice + + +class Dyson360Heurist(DysonVacuumDevice): + """Dyson 360 Heurist device.""" + + @property + def device_type(self) -> str: + """Return the device type.""" + return DEVICE_TYPE_360_HEURIST + + @property + def current_power_mode(self) -> VacuumHeuristPowerMode: + """Return current power mode.""" + return VacuumHeuristPowerMode(self._status["currentVacuumPowerMode"]) + + @property + def default_power_mode(self) -> VacuumHeuristPowerMode: + """Return default power mode.""" + return VacuumHeuristPowerMode(self._status["defaultVacuumPowerMode"]) + + @property + def current_cleaning_mode(self) -> CleaningMode: + """Return current cleaning mode.""" + return CleaningMode(self._status["currentCleaningMode"]) + + @property + def default_cleaning_mode(self) -> CleaningMode: + """Return default cleaning mode.""" + return CleaningMode(self._status["defaultCleaningMode"]) + + @property + def is_bin_full(self) -> bool: + """Return if the bin is full.""" + airways = self._status.get("faults", {}).get("AIRWAYS") + if airways is None: + return False + return ( + airways.get("active") is True and airways.get("description") == "1.0.-1" + ) # Not sure what this means + + def _send_command(self, command: str, data: Optional[dict] = None): + if data is None: + data = {} + data["mode-reason"] = "LAPP" + super()._send_command(command, data) + + def start_all_zones(self) -> None: + """Start cleaning of all zones.""" + self._send_command( + "START", {"cleaningMode": "global", "fullCleanType": "immediate"} + ) + + def set_default_power_mode(self, power_mode: VacuumHeuristPowerMode) -> None: + """Set default power mode.""" + self._send_command( + "STATE-SET", + {"defaults": {"defaultVacuumPowerMode": power_mode.value}}, + ) diff --git a/custom_components/dyson_cloud/vendor/libdyson/dyson_device.py b/custom_components/dyson_cloud/vendor/libdyson/dyson_device.py new file mode 100644 index 0000000..0760383 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/dyson_device.py @@ -0,0 +1,458 @@ +"""Dyson device.""" +from abc import abstractmethod +import json +import logging +import threading +from typing import Any, Optional, List, Dict, Union + +import paho.mqtt.client as mqtt + +from .const import ( + ENVIRONMENTAL_FAIL, + ENVIRONMENTAL_INIT, + ENVIRONMENTAL_OFF, + MessageType, +) +from .exceptions import ( + DysonConnectionRefused, + DysonConnectTimeout, + DysonInvalidCredential, + DysonNotConnected, DysonNoEnvironmentalData, +) +from .utils import mqtt_time + +_LOGGER = logging.getLogger(__name__) + +TIMEOUT = 10 + + +class DysonDevice: + """Base class for dyson devices.""" + + def __init__(self, serial: str, credential: str): + """Initialize the device.""" + self._serial = serial + self._credential = credential + self._mqtt_client = None + self._connected = threading.Event() + self._disconnected = threading.Event() + self._status = None + self._status_data_available = threading.Event() + self._callbacks = [] + + @property + def serial(self) -> str: + """Return the serial number of the device.""" + return self._serial + + @property + def is_connected(self) -> bool: + """Whether MQTT connection is active.""" + return self._connected.is_set() + + @property + @abstractmethod + def device_type(self) -> str: + """Device type.""" + + @property + @abstractmethod + def _status_topic(self) -> str: + """MQTT status topic.""" + + @property + def _command_topic(self) -> str: + """MQTT command topic.""" + return f"{self.device_type}/{self._serial}/command" + + def _request_first_data(self) -> bool: + """Request and wait for first data.""" + self.request_current_status() + return self._status_data_available.wait(timeout=TIMEOUT) + + def connect(self, host: str) -> None: + """Connect to the device MQTT broker.""" + self._disconnected.clear() + self._mqtt_client = mqtt.Client(protocol=mqtt.MQTTv31) + self._mqtt_client.username_pw_set(self._serial, self._credential) + error = None + + def _on_connect(client: mqtt.Client, userdata: Any, flags, rc): + _LOGGER.debug("Connected with result code %d", rc) + nonlocal error + if rc == mqtt.CONNACK_REFUSED_BAD_USERNAME_PASSWORD: + error = DysonInvalidCredential + elif rc != mqtt.CONNACK_ACCEPTED: + error = DysonConnectionRefused + else: + client.subscribe(self._status_topic) + self._connected.set() + + def _on_disconnect(client, userdata, rc): + _LOGGER.debug(f"Disconnected with result code {str(rc)}") + + self._disconnected.set() + + self._mqtt_client.on_connect = _on_connect + self._mqtt_client.on_disconnect = _on_disconnect + self._mqtt_client.on_message = self._on_message + self._mqtt_client.connect_async(host) + self._mqtt_client.loop_start() + if self._connected.wait(timeout=TIMEOUT): + if error is not None: + self.disconnect() + raise error + + _LOGGER.info("Connected to device %s", self._serial) + if self._request_first_data(): + self._mqtt_client.on_connect = self._on_connect + self._mqtt_client.on_disconnect = self._on_disconnect + return + + # Close connection if timeout or connected but failed to get data + self.disconnect() + + raise DysonConnectTimeout + + def disconnect(self) -> None: + """Disconnect from the device.""" + self._connected.clear() + self._mqtt_client.disconnect() + if not self._disconnected.wait(timeout=TIMEOUT): + _LOGGER.warning("Disconnect timed out") + self._mqtt_client.loop_stop() + self._mqtt_client = None + + def add_message_listener(self, callback) -> None: + """Add a callback to receive update notification.""" + self._callbacks.append(callback) + + def remove_message_listener(self, callback) -> None: + """Remove an existed callback.""" + if callback in self._callbacks: + self._callbacks.remove(callback) + + def _on_connect(self, client: mqtt.Client, userdata: Any, flags, rc): + _LOGGER.debug("Connected with result code %d", rc) + self._disconnected.clear() + self._connected.set() + client.subscribe(self._status_topic) + for callback in self._callbacks: + callback(MessageType.STATE) + + def _on_disconnect(self, client, userdata, rc): + _LOGGER.debug(f"Disconnected with result code {str(rc)}") + self._connected.clear() + self._disconnected.set() + for callback in self._callbacks: + callback(MessageType.STATE) + + def _on_message(self, client, userdata: Any, msg: mqtt.MQTTMessage): + payload = json.loads(msg.payload.decode("utf-8")) + self._handle_message(payload) + + def _handle_message(self, payload: dict) -> None: + if payload["msg"] in ["CURRENT-STATE", "STATE-CHANGE"]: + _LOGGER.debug("New state: %s", payload) + self._update_status(payload) + if not self._status_data_available.is_set(): + self._status_data_available.set() + for callback in self._callbacks: + callback(MessageType.STATE) + + @abstractmethod + def _update_status(self, payload: dict) -> None: + """Update the device status.""" + + def _send_command(self, command: str, data: Optional[dict] = None) -> None: + if not self.is_connected: + raise DysonNotConnected + if data is None: + data = {} + payload = { + "msg": command, + "time": mqtt_time(), + } + payload.update(data) + self._mqtt_client.publish(self._command_topic, json.dumps(payload)) + + def request_current_status(self) -> None: + """Request current status.""" + if not self.is_connected: + raise DysonNotConnected + payload = { + "msg": "REQUEST-CURRENT-STATE", + "time": mqtt_time(), + } + self._mqtt_client.publish(self._command_topic, json.dumps(payload)) + + +class DysonFanDevice(DysonDevice): + """Dyson fan device.""" + + def __init__(self, serial: str, credential: str, device_type: str): + """Initialize the device.""" + super().__init__(serial, credential) + self._device_type = device_type + + self._environmental_data = {} + self._environmental_data_available = threading.Event() + + @property + def device_type(self) -> str: + """Device type.""" + return self._device_type + + @property + def _status_topic(self) -> str: + """MQTT status topic.""" + return f"{self.device_type}/{self._serial}/status/current" + + @property + def fan_state(self) -> bool: + """Return if the fan is running.""" + return self._get_field_value(self._status, "fnst") == "FAN" + + @property + def speed(self) -> Optional[int]: + """Return fan speed.""" + speed = self._get_field_value(self._status, "fnsp") + if speed == "AUTO": + return None + return int(speed) + + @property + @abstractmethod + def is_on(self) -> bool: + """Return if the device is on.""" + + @property + @abstractmethod + def auto_mode(self) -> bool: + """Return auto mode status.""" + + @property + @abstractmethod + def oscillation(self) -> bool: + """Return oscillation status.""" + + @property + def night_mode(self) -> bool: + """Return night mode status.""" + return self._get_field_value(self._status, "nmod") == "ON" + + @property + def continuous_monitoring(self) -> bool: + """Return standby monitoring status.""" + return self._get_field_value(self._status, "rhtm") == "ON" + + @property + def error_code(self) -> str: + """Return error code.""" + return self._get_field_value(self._status, "ercd") + + @property + def warning_code(self) -> str: + """Return warning code.""" + return self._get_field_value(self._status, "wacd") + + @property + def formaldehyde(self) -> Optional[int]: + """Return formaldehyde reading.""" + val = self._get_environmental_field_value("hcho") + if val is None: + return None + + return int(val) + + @property + def humidity(self) -> int: + """Return humidity in percentage.""" + return self._get_environmental_field_value("hact") + + @property + def temperature(self) -> int: + """Return temperature in kelvin.""" + return self._get_environmental_field_value("tact", divisor=10) + + @property + @abstractmethod + def volatile_organic_compounds(self) -> int: + """Return VOCs.""" + + @property + def sleep_timer(self) -> int: + """Return sleep timer in minutes.""" + return self._get_environmental_field_value("sltm") + + @staticmethod + def _get_field_value(state: Dict[str, Any], field: str): + try: + return state[field][1] if isinstance(state[field], list) else state[field] + except: + return None + + def _get_environmental_field_value(self, field, divisor=1) -> Optional[Union[int, float]]: + value = self._get_field_value(self._environmental_data, field) + if value == "OFF" or value == "off": + return ENVIRONMENTAL_OFF + if value == "INIT": + return ENVIRONMENTAL_INIT + if value == "FAIL": + return ENVIRONMENTAL_FAIL + if value == "NONE" or value is None: + return None + if divisor == 1: + return int(value) + return float(value) / divisor + + def _handle_message(self, payload: dict) -> None: + super()._handle_message(payload) + if payload["msg"] == "ENVIRONMENTAL-CURRENT-SENSOR-DATA": + _LOGGER.debug("New environmental state: %s", payload) + self._environmental_data = payload["data"] + if not self._environmental_data_available.is_set(): + self._environmental_data_available.set() + for callback in self._callbacks: + callback(MessageType.ENVIRONMENTAL) + + def _update_status(self, payload: dict) -> None: + self._status = payload["product-state"] + + def _set_configuration(self, **kwargs: dict) -> None: + if not self.is_connected: + raise DysonNotConnected + payload = json.dumps( + { + "msg": "STATE-SET", + "time": mqtt_time(), + "mode-reason": "LAPP", + "data": kwargs, + } + ) + self._mqtt_client.publish(self._command_topic, payload, 1) + + def _request_first_data(self) -> bool: + """Request and wait for first data.""" + self.request_current_status() + self.request_environmental_data() + status_available = self._status_data_available.wait(timeout=TIMEOUT) + environmental_available = self._environmental_data_available.wait( + timeout=TIMEOUT + ) + return status_available and environmental_available + + def request_environmental_data(self): + """Request environmental sensor data.""" + if not self.is_connected: + raise DysonNotConnected + payload = { + "msg": "REQUEST-PRODUCT-ENVIRONMENT-CURRENT-SENSOR-DATA", + "time": mqtt_time(), + } + self._mqtt_client.publish(self._command_topic, json.dumps(payload)) + + @abstractmethod + def turn_on(self) -> None: + """Turn on the device.""" + + @abstractmethod + def turn_off(self) -> None: + """Turn off the device.""" + + def set_speed(self, speed: int) -> None: + """Set manual speed.""" + if not 1 <= speed <= 10: + raise ValueError("Invalid speed %s", speed) + self._set_speed(speed) + + @abstractmethod + def _set_speed(self, speed: int) -> None: + """Actually set the speed without range check.""" + + @abstractmethod + def enable_auto_mode(self) -> None: + """Turn on auto mode.""" + + @abstractmethod + def disable_auto_mode(self) -> None: + """Turn off auto mode.""" + + @abstractmethod + def enable_oscillation(self) -> None: + """Turn on oscillation.""" + + @abstractmethod + def disable_oscillation(self) -> None: + """Turn off oscillation.""" + + def enable_night_mode(self) -> None: + """Turn on auto mode.""" + self._set_configuration(nmod="ON") + + def disable_night_mode(self) -> None: + """Turn off auto mode.""" + self._set_configuration(nmod="OFF") + + @abstractmethod + def enable_continuous_monitoring(self) -> None: + """Turn on continuous monitoring.""" + + @abstractmethod + def disable_continuous_monitoring(self) -> None: + """Turn off continuous monitoring.""" + + def set_sleep_timer(self, duration: int) -> None: + """Set sleep timer.""" + if not 0 < duration <= 540: + raise ValueError("Duration must be between 1 and 540") + self._set_configuration(sltm="%04d" % duration) + + def disable_sleep_timer(self) -> None: + """Disable sleep timer.""" + self._set_configuration(sltm="OFF") + + def reset_filter(self) -> None: + """Reset filter life.""" + self._set_configuration(rstf="RSTF") + + +class DysonHeatingDevice(DysonFanDevice): + """Dyson heating fan device.""" + + @property + def focus_mode(self) -> bool: + """Return if fan focus mode is on.""" + return self._get_field_value(self._status, "ffoc") == "ON" + + @property + def heat_target(self) -> float: + """Return heat target in kelvin.""" + return int(self._get_field_value(self._status, "hmax")) / 10 + + @property + def heat_mode_is_on(self) -> bool: + """Return if heat mode is set to on.""" + return self._get_field_value(self._status, "hmod") == "HEAT" + + @property + def heat_status_is_on(self) -> bool: + """Return if the device is currently heating.""" + return self._get_field_value(self._status, "hsta") == "HEAT" + + def set_heat_target(self, heat_target: float) -> None: + """Set heat target in kelvin.""" + if not 274 <= heat_target <= 310: + raise ValueError("Heat target must be between 274 and 310 kelvin") + self._set_configuration( + hmod="HEAT", + hmax=f"{round(heat_target * 10):04d}", + ) + + def enable_heat_mode(self) -> None: + """Enable heat mode.""" + self._set_configuration(hmod="HEAT") + + def disable_heat_mode(self) -> None: + """Disable heat mode.""" + self._set_configuration(hmod="OFF") diff --git a/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_cool.py b/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_cool.py new file mode 100644 index 0000000..5fbf9f7 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_cool.py @@ -0,0 +1,177 @@ +"""Dyson Pure Cool fan.""" + +from abc import abstractmethod +from typing import Optional + +from .dyson_device import DysonFanDevice + + +class DysonPureCoolBase(DysonFanDevice): + """Dyson Pure Cool series base class.""" + + @property + def is_on(self) -> bool: + """Return if the device is on.""" + return self._get_field_value(self._status, "fpwr") == "ON" + + @property + def auto_mode(self) -> bool: + """Return auto mode status.""" + return self._get_field_value(self._status, "auto") == "ON" + + @property + @abstractmethod + def oscillation(self) -> bool: + """Return oscillation status.""" + + @property + def oscillation_status(self) -> bool: + """Return the status of oscillation.""" + return self._get_field_value(self._status, "oscs") == "ON" + + @property + def front_airflow(self) -> bool: + """Return if airflow from front is on.""" + return self._get_field_value(self._status, "fdir") == "ON" + + @property + def night_mode_speed(self) -> int: + """Return speed in night mode.""" + return int(self._get_field_value(self._status, "nmdv")) + + @property + def carbon_filter_life(self) -> Optional[int]: + """Return carbon filter life in percentage.""" + filter_life = self._get_field_value(self._status, "cflr") + if filter_life == "INV": + return None + return int(filter_life) + + @property + def hepa_filter_life(self) -> Optional[int]: + """Return HEPA filter life in percentage.""" + return int(self._get_field_value(self._status, "hflr")) + + @property + def particulate_matter_2_5(self): + """Return PM 2.5 in micro grams per cubic meter.""" + return int(self._get_environmental_field_value("pm25")) + + @property + def particulate_matter_10(self): + """Return PM 2.5 in micro grams per cubic meter.""" + return int(self._get_environmental_field_value("pm10")) + + @property + def volatile_organic_compounds(self) -> float: + """Return the index value for VOC""" + return self._get_environmental_field_value("va10", divisor=10) + + @property + def nitrogen_dioxide(self) -> float: + """Return the index value for nitrogen.""" + return self._get_environmental_field_value("noxl", divisor=10) + + def turn_on(self) -> None: + """Turn on the device.""" + self._set_configuration(fpwr="ON") + + def turn_off(self) -> None: + """Turn off the device.""" + self._set_configuration(fpwr="OFF") + + def _set_speed(self, speed: int) -> None: + self._set_configuration(fpwr="ON", fnsp=f"{speed:04d}") + + def enable_auto_mode(self) -> None: + """Turn on auto mode.""" + self._set_configuration(auto="ON") + + def disable_auto_mode(self) -> None: + """Turn off auto mode.""" + self._set_configuration(auto="OFF") + + def enable_continuous_monitoring(self) -> None: + """Turn on continuous monitoring.""" + self._set_configuration( + fpwr="ON" if self.is_on else "OFF", # Not sure about this + rhtm="ON", + ) + + def disable_continuous_monitoring(self) -> None: + """Turn off continuous monitoring.""" + self._set_configuration( + fpwr="ON" if self.is_on else "OFF", + rhtm="OFF", + ) + + def enable_front_airflow(self) -> None: + """Turn on front airflow.""" + self._set_configuration(fdir="ON") + + def disable_front_airflow(self) -> None: + """Turn off front airflow.""" + self._set_configuration(fdir="OFF") + + +class DysonPureCool(DysonPureCoolBase): + """Dyson Pure Cool device.""" + + @property + def oscillation(self) -> bool: + """Return oscillation status.""" + # Seems some devices use OION/OIOF while others uses ON/OFF + # https://github.com/shenxn/ha-dyson/issues/22 + return self._get_field_value(self._status, "oson") in ["OION", "ON"] + + @property + def oscillation_angle_low(self) -> int: + """Return oscillation low angle.""" + return int(self._get_field_value(self._status, "osal")) + + @property + def oscillation_angle_high(self) -> int: + """Return oscillation high angle.""" + return int(self._get_field_value(self._status, "osau")) + + def enable_oscillation( + self, + angle_low: Optional[int] = None, + angle_high: Optional[int] = None, + ) -> None: + """Turn on oscillation.""" + if angle_low is None: + angle_low = self.oscillation_angle_low + if angle_high is None: + angle_high = self.oscillation_angle_high + + if not 5 <= angle_low <= 355: + raise ValueError("angle_low must be between 5 and 355") + if not 5 <= angle_high <= 355: + raise ValueError("angle_high must be between 5 and 355") + if angle_low != angle_high and angle_low + 30 > angle_high: + raise ValueError( + "angle_high must be either equal to angle_low or at least 30 larger than angle_low" + ) + + current_oscillation_raw = self._get_field_value(self._status, "oson") + if current_oscillation_raw in ["OION", "OIOF"]: + oson = "OION" + else: + oson = "ON" + self._set_configuration( + oson=oson, + fpwr="ON", + ancp="CUST", + osal=f"{angle_low:04d}", + osau=f"{angle_high:04d}", + ) + + def disable_oscillation(self) -> None: + """Turn off oscillation.""" + current_oscillation_raw = self._get_field_value(self._status, "oson") + if current_oscillation_raw in ["OION", "OIOF"]: + oson = "OIOF" + else: + oson = "OFF" + self._set_configuration(oson=oson) diff --git a/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_cool_link.py b/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_cool_link.py new file mode 100644 index 0000000..ecf78e4 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_cool_link.py @@ -0,0 +1,93 @@ +"""Dyson Pure Cool Link fan.""" + +from .const import AirQualityTarget +from .dyson_device import DysonFanDevice + + +class DysonPureCoolLink(DysonFanDevice): + """Dyson Pure Cool Link device.""" + + @property + def fan_mode(self) -> str: + """Return the fan mode of the fan.""" + return self._get_field_value(self._status, "fmod") + + @property + def is_on(self) -> bool: + """Return if the device is on.""" + return self.fan_mode in ["FAN", "AUTO"] + + @property + def auto_mode(self) -> bool: + """Return auto mode status.""" + return self.fan_mode == "AUTO" + + @property + def oscillation(self) -> bool: + """Return oscillation status.""" + return self._get_field_value(self._status, "oson") == "ON" + + @property + def air_quality_target(self) -> AirQualityTarget: + """Return air quality target.""" + return AirQualityTarget(self._get_field_value(self._status, "qtar")) + + @property + def filter_life(self) -> int: + """Return filter life in hours.""" + return int(self._get_field_value(self._status, "filf")) + + @property + def particulates(self) -> int: + """Return particulate matter in unknown unit.""" + return self._get_environmental_field_value("pact") + + @property + def volatile_organic_compounds(self) -> int: + """Return VOCs in unknown unit.""" + return self._get_environmental_field_value("vact") + + def turn_on(self) -> None: + """Turn on the device.""" + self._set_configuration(fmod="FAN") + + def turn_off(self) -> None: + """Turn off the device.""" + self._set_configuration(fmod="OFF") + + def _set_speed(self, speed: int) -> None: + self._set_configuration(fmod="FAN", fnsp=f"{speed:04d}") + + def enable_auto_mode(self) -> None: + """Turn on auto mode.""" + self._set_configuration(fmod="AUTO") + + def disable_auto_mode(self) -> None: + """Turn off auto mode.""" + self._set_configuration(fmod="FAN") + + def enable_oscillation(self) -> None: + """Turn on oscillation.""" + self._set_configuration(oson="ON") + + def disable_oscillation(self) -> None: + """Turn off oscillation.""" + self._set_configuration(oson="OFF") + + def enable_continuous_monitoring(self) -> None: + """Turn on continuous monitoring.""" + self._set_configuration( + fmod=self.fan_mode, # Seems fmod is required to make this work + rhtm="ON", + ) + + def disable_continuous_monitoring(self) -> None: + """Turn off continuous monitoring.""" + self._set_configuration( + fmod=self.fan_mode, + rhtm="OFF", + ) + + def set_air_quality_target(self, air_quality_target: AirQualityTarget) -> None: + """Set air quality target.""" + self._set_configuration(qtar=air_quality_target.value) diff --git a/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_hot_cool.py b/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_hot_cool.py new file mode 100644 index 0000000..3ceb789 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_hot_cool.py @@ -0,0 +1,8 @@ +"""Dyson Pure Hot+Cool device.""" + +from .dyson_device import DysonHeatingDevice +from .dyson_pure_cool import DysonPureCool + + +class DysonPureHotCool(DysonPureCool, DysonHeatingDevice): + """Dyson Pure Hot+Cool device.""" diff --git a/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_hot_cool_link.py b/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_hot_cool_link.py new file mode 100644 index 0000000..c855dd2 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_hot_cool_link.py @@ -0,0 +1,21 @@ +"""Dyson Pure Hot+Cool Link device.""" + +from .dyson_device import DysonHeatingDevice +from .dyson_pure_cool_link import DysonPureCoolLink + + +class DysonPureHotCoolLink(DysonPureCoolLink, DysonHeatingDevice): + """Dyson Pure Hot+Cool Link device.""" + + @property + def tilt(self) -> bool: + """Return tilt status.""" + return self._get_field_value(self._status, "tilt") == "TILT" + + def enable_focus_mode(self) -> None: + """Enable fan focus mode.""" + self._set_configuration(ffoc="ON") + + def disable_focus_mode(self) -> None: + """Disable fan focus mode.""" + self._set_configuration(ffoc="OFF") diff --git a/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_humidify_cool.py b/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_humidify_cool.py new file mode 100644 index 0000000..af3b91a --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/dyson_pure_humidify_cool.py @@ -0,0 +1,102 @@ +"""Dyson Pure Humidify+Cool device.""" + +from typing import Optional + +from .const import HumidifyOscillationMode, WaterHardness +from .dyson_pure_cool import DysonPureCoolBase + +WATER_HARDNESS_ENUM_TO_STR = { + WaterHardness.SOFT: "2025", + WaterHardness.MEDIUM: "1350", + WaterHardness.HARD: "0675", +} +WATER_HARDNESS_STR_TO_ENUM = { + str_: enum for enum, str_ in WATER_HARDNESS_ENUM_TO_STR.items() +} + + +class DysonPurifierHumidifyCool(DysonPureCoolBase): + """Dyson Pure Humidify+Cool device.""" + + @property + def oscillation(self) -> bool: + """Return oscillation status.""" + return self._get_field_value(self._status, "oson") == "ON" + + @property + def oscillation_mode(self) -> HumidifyOscillationMode: + """Return oscillation mode.""" + return HumidifyOscillationMode(self._get_field_value(self._status, "ancp")) + + @property + def humidification(self) -> bool: + """Return if humidification is on.""" + return self._get_field_value(self._status, "hume") == "HUMD" + + @property + def humidification_auto_mode(self) -> bool: + """Return if humidification auto mode is on.""" + return self._get_field_value(self._status, "haut") == "ON" + + @property + def target_humidity(self) -> int: + """Return target humidity in percentage.""" + return int(self._get_field_value(self._status, "humt")) + + @property + def auto_target_humidity(self) -> int: + """Return humidification auto mode target humidity.""" + return int(self._get_field_value(self._status, "rect")) + + @property + def water_hardness(self) -> WaterHardness: + """Return the water hardness setting.""" + return WATER_HARDNESS_STR_TO_ENUM[self._get_field_value(self._status, "wath")] + + @property + def time_until_next_clean(self) -> int: + """Return the time remaining in hours before the next deep cleaning.""" + return int(self._get_field_value(self._status, "cltr")) + + @property + def clean_time_remaining(self) -> int: + """Return the time remaining in minutes before the cleaning finishes.""" + return int(self._get_field_value(self._status, "cdrr")) + + def enable_oscillation( + self, oscillation_mode: Optional[HumidifyOscillationMode] = None + ) -> None: + """Turn on oscillation.""" + if oscillation_mode is None: + oscillation_mode = self.oscillation_mode + + self._set_configuration(oson="ON", fpwr="ON", ancp=oscillation_mode.value) + + def disable_oscillation(self) -> None: + """Turn off oscillation.""" + self._set_configuration(oson="OFF") + + def enable_humidification(self) -> None: + """Enable humidification.""" + self._set_configuration(hume="HUMD") + + def disable_humidification(self) -> None: + """Disable humidification.""" + self._set_configuration(hume="OFF") + + def enable_humidification_auto_mode(self) -> None: + """Enable humidification auto mode.""" + self._set_configuration(haut="ON") + + def disable_humidification_auto_mode(self) -> None: + """Disable humidification auto mode.""" + self._set_configuration(haut="OFF") + + def set_target_humidity(self, target_humidity: int) -> None: + """Set target humidity.""" + self._set_configuration(humt=f"{target_humidity:04d}", haut="OFF") + + def set_water_hardness(self, water_hardness: WaterHardness) -> None: + """Set water hardness.""" + self._set_configuration(wath=WATER_HARDNESS_ENUM_TO_STR[water_hardness]) + diff --git a/custom_components/dyson_cloud/vendor/libdyson/dyson_vacuum_device.py b/custom_components/dyson_cloud/vendor/libdyson/dyson_vacuum_device.py new file mode 100644 index 0000000..3d7f44c --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/dyson_vacuum_device.py @@ -0,0 +1,80 @@ +"""Dyson vacuum device.""" + +from typing import Optional, Tuple + +from .const import CleaningType, VacuumState +from .dyson_device import DysonDevice + + +class DysonVacuumDevice(DysonDevice): + """Dyson vacuum device.""" + + @property + def _status_topic(self) -> str: + """MQTT status topic.""" + return f"{self.device_type}/{self._serial}/status" + + @property + def state(self) -> VacuumState: + """State of the device.""" + return VacuumState( + self._status["state"] + if "state" in self._status + else self._status["newstate"] + ) + + @property + def cleaning_type(self) -> Optional[CleaningType]: + """Return the type of the current cleaning task.""" + cleaning_type = self._status["fullCleanType"] + if cleaning_type == "": + return None + return CleaningType(cleaning_type) + + @property + def cleaning_id(self) -> Optional[str]: + """Return the id of the current cleaning task.""" + cleaning_id = self._status["cleanId"] + if cleaning_id == "": + return None + return cleaning_id + + @property + def battery_level(self) -> int: + """Battery level of the device in percentage.""" + return self._status["batteryChargeLevel"] + + @property + def position(self) -> Optional[Tuple[int, int]]: + """Position (x, y) of the device.""" + if ( + "globalPosition" in self._status + and len(self._status["globalPosition"]) == 2 + ): + return tuple(self._status["globalPosition"]) + return None + + @property + def is_charging(self) -> bool: + """Whether the device is charging.""" + return self.state in [ + VacuumState.INACTIVE_CHARGING, + VacuumState.INACTIVE_CHARGED, + VacuumState.FULL_CLEAN_CHARGING, + VacuumState.MAPPING_CHARGING, + ] + + def _update_status(self, payload: dict) -> None: + self._status = payload + + def pause(self) -> None: + """Pause cleaning.""" + self._send_command("PAUSE") + + def resume(self) -> None: + """Resume cleaning.""" + self._send_command("RESUME") + + def abort(self) -> None: + """Abort cleaning.""" + self._send_command("ABORT") diff --git a/custom_components/dyson_cloud/vendor/libdyson/exceptions.py b/custom_components/dyson_cloud/vendor/libdyson/exceptions.py new file mode 100644 index 0000000..1abeb27 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/exceptions.py @@ -0,0 +1,61 @@ +"""Dyson Python library exceptions.""" + + +class DysonException(Exception): + """Base class for exceptions.""" + + +class DysonNetworkError(DysonException): + """Represents network error.""" + + +class DysonServerError(DysonException): + """Represents Dyson server error.""" + + +class DysonInvalidAccountStatus(DysonException): + """Represents invalid account status.""" + + +class DysonLoginFailure(DysonException): + """Represents failure during logging in.""" + + +class DysonAPIProvisionFailure(DysonException): + """Represents failure during logging in.""" + + +class DysonOTPTooFrequently(DysonException): + """Represents requesting OTP code too frequently.""" + + +class DysonAuthRequired(DysonException): + """Represents not logged into could.""" + + +class DysonInvalidAuth(DysonException): + """Represents invalid authentication.""" + + +class DysonConnectTimeout(DysonException): + """Represents mqtt connection timeout.""" + + +class DysonNotConnected(DysonException): + """Represents mqtt not connected.""" + + +class DysonInvalidCredential(DysonException): + """Represents invalid mqtt credential.""" + + +class DysonConnectionRefused(DysonException): + """Represents mqtt connection refused by the server.""" + + +class DysonFailedToParseWifiInfo(DysonException): + """Represents failed to parse WiFi information.""" + + +class DysonNoEnvironmentalData(DysonException): + """Represents mqtt not connected.""" diff --git a/custom_components/dyson_cloud/vendor/libdyson/utils.py b/custom_components/dyson_cloud/vendor/libdyson/utils.py new file mode 100644 index 0000000..99045a3 --- /dev/null +++ b/custom_components/dyson_cloud/vendor/libdyson/utils.py @@ -0,0 +1,51 @@ +"""Utility functions for Dyson Python library.""" + +import base64 +import hashlib +import re +import time +from typing import Tuple + +from .const import DEVICE_TYPE_360_EYE +from .exceptions import DysonFailedToParseWifiInfo + +# For some devices, the model in WiFi SSID is not the same as the model for MQTT. +# The model on Dyson Cloud always matches the one used for MQTT. +_DEVICE_TYPE_MAP = { + "455A": "455", +} + + +def mqtt_time(): + """Return current time string for mqtt messages.""" + return time.strftime("%Y-%m-%dT%H:%M:%SZ", time.gmtime()) + + +def get_credential_from_wifi_password(wifi_password: str) -> str: + """Calculate MQTT credential from WiFi password.""" + hash_ = hashlib.sha512() + hash_.update(wifi_password.encode("utf-8")) + return base64.b64encode(hash_.digest()).decode("utf-8") + + +def get_mqtt_info_from_wifi_info( + wifi_ssid: str, wifi_password: str +) -> Tuple[str, str, str]: + """Get MQTT information from WiFi information.""" + result = re.match(r"^(360EYE-)?(?P[0-9A-Z]{3}-[A-Z]{2}-[0-9A-Z]{8,})$", wifi_ssid) + if result is not None: + serial = result.group("serial") + device_type = DEVICE_TYPE_360_EYE + else: + result = re.match( + r"^DYSON-([0-9A-Z]{3}-[A-Z]{2}-[0-9A-Z]{8,})-([0-9]{3}[A-Z]?)$", wifi_ssid + ) + if result is not None: + serial = result.group(1) + device_type = result.group(2) + device_type = _DEVICE_TYPE_MAP.get(device_type, device_type) + else: + raise DysonFailedToParseWifiInfo + + credential = get_credential_from_wifi_password(wifi_password) + return serial, credential, device_type