Skip to content

Commit

Permalink
Combine Cloud and Local versions of the Dyson Integration (#50)
Browse files Browse the repository at this point in the history
* Improving the readme.

* Updating config flow for more clarity

* Merge Dyson Cloud into ha-dyson
  • Loading branch information
dotvezz authored Jul 11, 2023
1 parent 73ea3e3 commit 5132321
Show file tree
Hide file tree
Showing 8 changed files with 396 additions and 57 deletions.
65 changes: 24 additions & 41 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,11 +1,6 @@
# HomeAssistant Custom Integration for Dyson

This custom integration is still under development.

This is a HA custom integration for dyson. There are several main differences between this custom integration and the official dyson integration:

- It does not rely on a dyson account. Which means once configured, the integration will no longer login to the Dyson cloud service so fast and more reliable start up process.
- Config flow and discovery is supported, so easier configuration.
This is a Home Assistant custom integration for dyson under active development.

## Migration from shenxn/ha-dyson

Expand All @@ -16,10 +11,9 @@ If you used the original repository from shenxn, you can migrate fairly easily:
This is less proven, but it is possible to switch over with zero impact to your current integration configuration, entities/devices, or dashboards. I don't know what side-effects it may have though (leftover old config data might start causing issues or something - no guarantees).

1. Remove the ha-dyson and ha-dyson-cloud custom repositories from HACS
- _Without_ removing the integrations themselves.
3. Add the new [ha-dyson](https://github.com/libdyson-wg/ha-dyson) and [ha-dyson-cloud](https://github.com/libdyson-wg/ha-dyson-cloud) custom repositories
- The ha-dyson-cloud repository is only necessary if you already use it, or are intending to use its features. It is not required, but currently, it makes setting up new devices like HP07 (527K) much simpler.
4. Update the ha-dyson and ha-dyson-cloud repositories using the HACS updater
- _Without_ removing the integrations themselves
3. Install the new [ha-dyson](https://github.com/libdyson-wg/ha-dyson)
- If you installed this as a Custom Repository, update the ha-dyson repository using the HACS updater

### Proven some-reconfiguration migration

Expand All @@ -28,67 +22,56 @@ This is proven to work without any side effects. If you used the default IDs for
1. Remove the Dyson Local and Dyson Cloud _integrations_ from your /config/integrations page.
1. Remove the Dyson Local and Dyson Cloud _integrations_ from your /hacs/integrations page.
2. Remove the dyson-ha and dyson-ha-cloud custom repositories from HACS
3. Add the new [dyson-ha](https://github.com/libdyson-wg/ha-dyson) and [dyson-ha-cloud](https://github.com/libdyson-wg/ha-dyson-cloud) custom repositories
- The libdyson-ha-cloud repository is only necessary if you already use it, or are intending to use its features. It is not required, but currently, it makes setting up new devices like HP07 (527K) much simpler.
4. Update the dyson-ha and dyson-ha-cloud repositories
3. Add the new [dyson-ha](https://github.com/libdyson-wg/ha-dyson)
- If you installed this as a Custom Repository, update the ha-dyson repository using the HACS updater

## Installation

The minimum supported Home Assistant version is 2021.12.0.

You can install using HACS. Adding https://github.com/libdyson-wg/ha-dyson as custom repository and then install Dyson Local. If you want cloud functionalities as well, add https://github.com/libdyson-wg/ha-dyson-cloud and install Dyson Cloud.

You can also install manually
You can install using HACS. If it is not yet available in the default HACS search, you can add https://github.com/libdyson-wg/ha-dyson as a custom repository.

## Local and Cloud
You can also install manually by copying the `custom_components` from this repository into your Home Assistant installation.

There are two integrations, Dyson Local and Dyson Cloud. Due to the limitation of HACS, they are split into two repositories. This repository hosts Dyson Local, and https://github.com/libdyson-wg/ha-dyson-cloud hosts Dyson Cloud.
### Dyson Devices Supported

### Dyson Local

Dyson Local uses MQTT-based protocol to communicate with local Dyson devices using credentials. Currently it supports
This integration uses MQTT-based protocol to communicate with Dyson devices. Only WiFi enabled models have this capability. Currently the following models are supported, and support for more models can be added on request.

- Dyson 360 Eye robot vacuum
- Dyson 360 Heurist robot vacuum
- Dyson Pure Cool
- Dyson Purifier Cool
- Dyson Purifier Cool Formaldehyde
- Dyson Pure Cool Desk
- Dyson Pure Cool Link
- Dyson Pure Cool Link Desk
- Dyson Pure Hot+Cool
- Dyson Purifier Hot+Cool
- Dyson Pure Hot+Cool Link
- Dyson Purifier Hot+Cool
- Dyson Purifier Hot+Cool Formaldehyde
- Dyson Pure Humidity+Cool
- Dyson Purifier Humidity+Cool
- Dyson Purifier Humidity+Cool Formaldehyde

### Dyson Cloud
### MyDyson Accounts

Dyson Cloud uses HTTP-based API to communicate with cloud service. Currently it supports getting device credentials and show all devices as discovered entities under the Integrations page. It also supports getting cleaning maps as `camera` entities for 360 Eye robot vacuum.
MyDyson mobile apps use an HTTP-based API, which is also used by the MyDyson part of this integration. Currently it supports automated setup of your devices by discovering and fetching credentials from the API. It also supports getting cleaning maps as `camera` entities for 360 Eye robot vacuum.

## Setup

### Setup using device WiFi information

Version 0.6.1 introduced a new way to set up. This is inspired by https://community.home-assistant.io/t/dyson-pure-cool-link-local-mqtt-control/217263. Set up through UI and select "Setup using WiFi information". Find your device WiFi SSID and password on the sticker on your device body or user's manual (See the figure below). Don't fill in your home WiFi information. Note that this method only uses SSID and password to calculate serial, credential, and device type so you still need to setup your device on the official mobile app first.

### Setup using Dyson cloud account
Note: Some new models released after 2020 do not ship with a Wi-Fi information sticker. They are still supported by this integration, but can only be configured via your MyDyson account. After setting up your devices, your account can be deleted from Home Assistant if you prefer to stay offline.

You can also set up Dyson Cloud first so that you don't need to manually get device credentials. To do so, go to **Configuration** -> **Integrations** and click the **+** button. Then find Dyson Cloud. After successful setup, all devices under the account will be shown as discovered entities and you can then set up Dyson Local with single click. Leave host blank to using zeroconf discovery. After that, you can even remove Dyson Cloud entity if you don't need cleaning maps. All local devices that are already set up will remain untouched.
Find your device WiFi SSID and password on the sticker on your device body or user's manual. Don't fill in your home WiFi information. Note that this method only uses SSID and password to calculate serial, credential, and device type so you still need to setup your device on the official mobile app first.

### Setup manually
### Setup using your MyDyson account

If you want to manually set up Dyson Local, you need to get credentials first. Clone or download https://github.com/libdyson-wg/libdyson-neon, then use `python3 get_devices.py` to do that. You may need to install some dependencies using `pip3 install -r requirements.txt`.
You can also set up a MyDyson account first so that you don't need to manually get device credentials. After successfully connecting your account, all devices under the account will be shown as discovered entities and you can easily set them up. After that, you can even remove MyDyson account entity if you don't need cleaning maps for the 360 Eye vacuum. All local devices that are already set up will remain untouched.

## Debug Log

To enable debug log, add the following lines to your `configuration.yaml` and restart your HomeAssistant.
### Setup manually

```yaml
logger:
default: info
logs:
libdyson: debug
custom_components.dyson_local: debug
custom_components.dyson_cloud: debug
```
If you want to manually set up a Dyson device, you need to get credentials first. Clone or download https://github.com/libdyson-wg/libdyson-neon, then use `python3 get_devices.py` to do that. You may need to install some dependencies using `pip3 install -r requirements.txt`.

## FAQ

Expand Down
57 changes: 55 additions & 2 deletions custom_components/dyson_local/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,12 +15,20 @@
MessageType,
get_device,
)
from .vendor.libdyson.cloud import (
DysonAccountCN,
DysonAccount,
)
from .vendor.libdyson.discovery import DysonDiscovery
from .vendor.libdyson.dyson_device import DysonDevice
from .vendor.libdyson.exceptions import DysonException
from .vendor.libdyson.exceptions import (
DysonException,
DysonNetworkError,
DysonLoginFailure,
)

from homeassistant.components.zeroconf import async_get_instance
from homeassistant.config_entries import ConfigEntry
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.exceptions import ConfigEntryNotReady
Expand All @@ -37,10 +45,19 @@
DOMAIN,
)

from .cloud.const import (
CONF_REGION,
CONF_AUTH,
DATA_ACCOUNT,
DATA_DEVICES,
)

_LOGGER = logging.getLogger(__name__)

ENVIRONMENTAL_DATA_UPDATE_INTERVAL = timedelta(seconds=30)

PLATFORMS = ["camera"]


async def async_setup(hass: HomeAssistant, config: dict) -> bool:
"""Set up Dyson integration."""
Expand All @@ -52,8 +69,44 @@ async def async_setup(hass: HomeAssistant, config: dict) -> bool:
return True


async def async_setup_account(hass: HomeAssistant, entry: ConfigEntry) -> bool:
"""Set up a MyDyson Account."""
if entry.data[CONF_REGION] == "CN":
account = DysonAccountCN(entry.data[CONF_AUTH])
else:
account = DysonAccount(entry.data[CONF_AUTH])
try:
devices = await hass.async_add_executor_job(account.devices)
except DysonNetworkError:
_LOGGER.error("Cannot connect to Dyson cloud service.")
raise ConfigEntryNotReady

for device in devices:
hass.async_create_task(
hass.config_entries.flow.async_init(
DOMAIN,
context={"source": SOURCE_DISCOVERY},
data=device,
)
)

hass.data[DOMAIN][entry.entry_id] = {
DATA_ACCOUNT: account,
DATA_DEVICES: devices,
}
for component in PLATFORMS:
hass.async_create_task(
hass.config_entries.async_forward_entry_setup(entry, component)
)

return True


async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
"""Set up Dyson from a config entry."""
if CONF_REGION in entry.data:
return await async_setup_account(hass, entry)

device = get_device(
entry.data[CONF_SERIAL],
entry.data[CONF_CREDENTIAL],
Expand Down
24 changes: 24 additions & 0 deletions custom_components/dyson_local/cloud/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
"""Support for Dyson cloud account."""

import asyncio
import logging
from functools import partial

from homeassistant.exceptions import ConfigEntryNotReady
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 ..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

_LOGGER = logging.getLogger(__name__)
99 changes: 99 additions & 0 deletions custom_components/dyson_local/cloud/camera.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,99 @@
"""Camera platform for Dyson cloud."""
from typing import Callable
import logging
from homeassistant.core import HomeAssistant
from homeassistant.config_entries import ConfigEntry
from homeassistant.components.camera import Camera
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

_LOGGER = logging.getLogger(__name__)

SCAN_INTERVAL = timedelta(minutes=30)


async def async_setup_entry(
hass: HomeAssistant, config_entry: ConfigEntry, async_add_entities: Callable
) -> None:
"""Set up Dyson fan from a config entry."""
data = hass.data[DOMAIN][config_entry.entry_id]
account = data[DATA_ACCOUNT]
devices = data[DATA_DEVICES]
entities = []
for device in devices:
if device.product_type not in [DEVICE_TYPE_360_EYE]:
continue
entities.append(DysonCleaningMapEntity(
DysonCloud360Eye(account, device.serial),
device,
))
async_add_entities(entities, True)


class DysonCleaningMapEntity(Camera):
"""Dyson vacuum cleaning map entity."""

def __init__(self, device: DysonCloud360Eye, device_info: DysonDeviceInfo):
super().__init__()
self._device = device
self._device_info = device_info
self._last_cleaning_task = None
self._image = None

@property
def name(self) -> str:
"""Return entity name."""
return f"{self._device_info.name} Cleaning Map"

@property
def unique_id(self) -> str:
"""Return entity unique id."""
return self._device_info.serial

@property
def device_info(self) -> dict:
"""Return device info of the entity."""
return {
"identifiers": {(DOMAIN, self._device_info.serial)},
"name": self._device_info.name,
"manufacturer": "Dyson",
"model": self._device_info.product_type,
"sw_version": self._device_info.version,
}

@property
def icon(self) -> str:
"""Return entity icon."""
return "mdi:map"

def camera_image(self):
"""Return cleaning map."""
return self._image

def update(self):
"""Check for map update."""
_LOGGER.debug("Running cleaning map update for %s", self._device_info.name)
cleaning_tasks = self._device.get_cleaning_history()

last_task = None
for task in cleaning_tasks:
if task.area > 0.0:
# Skip cleaning tasks with 0 area, map not available
last_task = task
break
if last_task is None:
_LOGGER.debug("No cleaning history found.")
self._last_cleaning_task = None
return

if last_task == self._last_cleaning_task:
_LOGGER.debug("Cleaning task not changed. Skip update.")
return
self._last_cleaning_task = last_task
self._image = self._device.get_cleaning_map(
self._last_cleaning_task.cleaning_id
)
7 changes: 7 additions & 0 deletions custom_components/dyson_local/cloud/const.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
DOMAIN = "dyson_cloud"

CONF_REGION = "region"
CONF_AUTH = "auth"

DATA_ACCOUNT = "account"
DATA_DEVICES = "devices"
Loading

0 comments on commit 5132321

Please sign in to comment.